From b274baf67e27d79fd4e764607ded7c5aa755ee8b Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 19 Jun 2024 09:06:48 -0500 Subject: [PATCH 01/45] unit test coverage for HostNameTableCell --- .../AccessKeyTable/HostNameTableCell.test.tsx | 115 ++++++++++++++++++ .../AccessKeyTable/HostNameTableCell.tsx | 17 +-- 2 files changed, 124 insertions(+), 8 deletions(-) create mode 100644 packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx diff --git a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx new file mode 100644 index 00000000000..92c072ca4e9 --- /dev/null +++ b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx @@ -0,0 +1,115 @@ +import '@testing-library/jest-dom'; +import { waitFor } from '@testing-library/react'; +import React from 'react'; + +import { regionFactory } from 'src/factories'; +import { makeResourcePage } from 'src/mocks/serverHandlers'; +import { HttpResponse, http, server } from 'src/mocks/testServer'; +import { renderWithThemeAndHookFormContext } from 'src/utilities/testHelpers'; + +import { HostNameTableCell } from './HostNameTableCell'; + +const storageKeyData = { + access_key: 'test_key', + bucket_access: null, + id: 12345, + label: 'this is regular key', + limited: false, + regions: [ + { + id: 'us-east', + s3_endpoint: 'alpha.test.com', + }, + ], + secret_key: '[test]', +}; + +describe('HostNameTableCell', () => { + it('should render "None" when there are no regions', () => { + const { getByText } = renderWithThemeAndHookFormContext({ + component: ( + + ), + }); + + expect(getByText('None')).toBeInTheDocument(); + }); + + test('should render "Regions/S3 Hostnames" cell when there are regions', async () => { + const region = regionFactory.build({ + capabilities: ['Object Storage'], + id: 'us-east', + label: 'Newark, NJ', + }); + + server.use( + http.get('*/v4/regions', () => { + return HttpResponse.json(makeResourcePage([region])); + }) + ); + const { findByText } = renderWithThemeAndHookFormContext({ + component: ( + + ), + }); + + const hostname = await findByText('Newark, NJ: alpha.test.com'); + + await waitFor(() => expect(hostname).toBeInTheDocument()); + }); + test('should render all "Regions/S3 Hostnames" in the cell when there are multiple regions', async () => { + const region = regionFactory.build({ + capabilities: ['Object Storage'], + id: 'us-east', + label: 'Newark, NJ', + }); + + server.use( + http.get('*/v4/regions', () => { + return HttpResponse.json(makeResourcePage([region])); + }) + ); + const { findByText } = renderWithThemeAndHookFormContext({ + component: ( + + ), + }); + const hostname = await findByText('Newark, NJ: alpha.test.com'); + const moreButton = await findByText(/and\s+1\s+more\.\.\./); + await waitFor(() => expect(hostname).toBeInTheDocument()); + + await expect(moreButton).toBeInTheDocument(); + }); +}); diff --git a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx index e5fb3ce88db..644e2558d5d 100644 --- a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx +++ b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx @@ -1,7 +1,3 @@ -import { - ObjectStorageKey, - RegionS3EndpointAndID, -} from '@linode/api-v4/lib/object-storage'; import { styled } from '@mui/material/styles'; import React from 'react'; @@ -11,6 +7,11 @@ import { TableCell } from 'src/components/TableCell'; import { useRegionsQuery } from 'src/queries/regions/regions'; import { getRegionsByRegionId } from 'src/utilities/regions'; +import type { + ObjectStorageKey, + RegionS3EndpointAndID, +} from '@linode/api-v4/lib/object-storage'; + type Props = { setHostNames: (hostNames: RegionS3EndpointAndID[]) => void; setShowHostNamesDrawers: (show: boolean) => void; @@ -31,14 +32,14 @@ export const HostNameTableCell = ({ if (!regionsLookup || !regionsData || !regions || regions.length === 0) { return None; } + const label = regionsLookup[storageKeyData.regions[0].id]?.label; + const s3endPoint = storageKeyData?.regions[0]?.s3_endpoint; return ( - {`${regionsLookup[storageKeyData.regions[0].id].label}: ${ - storageKeyData?.regions[0]?.s3_endpoint - } `} + {`${label}: ${s3endPoint} `} {storageKeyData?.regions?.length === 1 && ( - + )} {storageKeyData.regions.length > 1 && ( Date: Wed, 19 Jun 2024 09:09:10 -0500 Subject: [PATCH 02/45] Revert "unit test coverage for HostNameTableCell" This reverts commit b274baf67e27d79fd4e764607ded7c5aa755ee8b. --- .../AccessKeyTable/HostNameTableCell.test.tsx | 115 ------------------ .../AccessKeyTable/HostNameTableCell.tsx | 17 ++- 2 files changed, 8 insertions(+), 124 deletions(-) delete mode 100644 packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx diff --git a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx deleted file mode 100644 index 92c072ca4e9..00000000000 --- a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.test.tsx +++ /dev/null @@ -1,115 +0,0 @@ -import '@testing-library/jest-dom'; -import { waitFor } from '@testing-library/react'; -import React from 'react'; - -import { regionFactory } from 'src/factories'; -import { makeResourcePage } from 'src/mocks/serverHandlers'; -import { HttpResponse, http, server } from 'src/mocks/testServer'; -import { renderWithThemeAndHookFormContext } from 'src/utilities/testHelpers'; - -import { HostNameTableCell } from './HostNameTableCell'; - -const storageKeyData = { - access_key: 'test_key', - bucket_access: null, - id: 12345, - label: 'this is regular key', - limited: false, - regions: [ - { - id: 'us-east', - s3_endpoint: 'alpha.test.com', - }, - ], - secret_key: '[test]', -}; - -describe('HostNameTableCell', () => { - it('should render "None" when there are no regions', () => { - const { getByText } = renderWithThemeAndHookFormContext({ - component: ( - - ), - }); - - expect(getByText('None')).toBeInTheDocument(); - }); - - test('should render "Regions/S3 Hostnames" cell when there are regions', async () => { - const region = regionFactory.build({ - capabilities: ['Object Storage'], - id: 'us-east', - label: 'Newark, NJ', - }); - - server.use( - http.get('*/v4/regions', () => { - return HttpResponse.json(makeResourcePage([region])); - }) - ); - const { findByText } = renderWithThemeAndHookFormContext({ - component: ( - - ), - }); - - const hostname = await findByText('Newark, NJ: alpha.test.com'); - - await waitFor(() => expect(hostname).toBeInTheDocument()); - }); - test('should render all "Regions/S3 Hostnames" in the cell when there are multiple regions', async () => { - const region = regionFactory.build({ - capabilities: ['Object Storage'], - id: 'us-east', - label: 'Newark, NJ', - }); - - server.use( - http.get('*/v4/regions', () => { - return HttpResponse.json(makeResourcePage([region])); - }) - ); - const { findByText } = renderWithThemeAndHookFormContext({ - component: ( - - ), - }); - const hostname = await findByText('Newark, NJ: alpha.test.com'); - const moreButton = await findByText(/and\s+1\s+more\.\.\./); - await waitFor(() => expect(hostname).toBeInTheDocument()); - - await expect(moreButton).toBeInTheDocument(); - }); -}); diff --git a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx index 644e2558d5d..e5fb3ce88db 100644 --- a/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx +++ b/packages/manager/src/features/ObjectStorage/AccessKeyLanding/AccessKeyTable/HostNameTableCell.tsx @@ -1,3 +1,7 @@ +import { + ObjectStorageKey, + RegionS3EndpointAndID, +} from '@linode/api-v4/lib/object-storage'; import { styled } from '@mui/material/styles'; import React from 'react'; @@ -7,11 +11,6 @@ import { TableCell } from 'src/components/TableCell'; import { useRegionsQuery } from 'src/queries/regions/regions'; import { getRegionsByRegionId } from 'src/utilities/regions'; -import type { - ObjectStorageKey, - RegionS3EndpointAndID, -} from '@linode/api-v4/lib/object-storage'; - type Props = { setHostNames: (hostNames: RegionS3EndpointAndID[]) => void; setShowHostNamesDrawers: (show: boolean) => void; @@ -32,14 +31,14 @@ export const HostNameTableCell = ({ if (!regionsLookup || !regionsData || !regions || regions.length === 0) { return None; } - const label = regionsLookup[storageKeyData.regions[0].id]?.label; - const s3endPoint = storageKeyData?.regions[0]?.s3_endpoint; return ( - {`${label}: ${s3endPoint} `} + {`${regionsLookup[storageKeyData.regions[0].id].label}: ${ + storageKeyData?.regions[0]?.s3_endpoint + } `} {storageKeyData?.regions?.length === 1 && ( - + )} {storageKeyData.regions.length > 1 && ( Date: Thu, 26 Sep 2024 14:38:58 -0400 Subject: [PATCH 03/45] chore: [M3-8662] - Update Github Actions actions (#11009) * update actions * add changeset --------- Co-authored-by: Banks Nussman --- .github/workflows/ci.yml | 161 ++++++++---------- .github/workflows/coverage.yml | 16 +- .github/workflows/coverage_badge.yml | 6 +- .github/workflows/coverage_comment.yml | 4 +- .github/workflows/docs.yml | 10 +- .github/workflows/e2e_schedule_and_push.yml | 8 +- .github/workflows/security_scan.yml | 2 +- .../pr-11009-tech-stories-1727310949305.md | 5 + 8 files changed, 102 insertions(+), 110 deletions(-) create mode 100644 packages/manager/.changeset/pr-11009-tech-stories-1727310949305.md diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 014410c88cb..962897a5748 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -15,11 +15,11 @@ jobs: package: ["linode-manager", "@linode/api-v4", "@linode/validation"] runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules @@ -30,35 +30,32 @@ jobs: build-validation: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - run: yarn --frozen-lockfile - run: yarn workspace @linode/validation run build - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: name: packages-validation-lib - path: | - packages/validation/index.js - packages/validation/lib + path: packages/validation/lib publish-validation: runs-on: ubuntu-latest if: github.ref == 'refs/heads/master' - needs: - - build-validation + needs: build-validation steps: - - uses: actions/checkout@v2 - - uses: actions/download-artifact@v2 + - uses: actions/checkout@v4 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation + path: packages/validation/lib - uses: JS-DevTools/npm-publish@v1 id: npm-publish with: @@ -80,69 +77,64 @@ jobs: runs-on: ubuntu-latest needs: build-validation steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - run: yarn --frozen-lockfile - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation + path: packages/validation/lib - run: yarn workspace @linode/api-v4 run test build-sdk: runs-on: ubuntu-latest - needs: - - build-validation + needs: build-validation steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation + path: packages/validation/lib - run: yarn --frozen-lockfile - run: yarn workspace @linode/api-v4 run build - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: name: packages-api-v4-lib - path: | - packages/api-v4/index.js - packages/api-v4/index.node.js - packages/api-v4/lib + path: packages/api-v4/lib validate-sdk: runs-on: ubuntu-latest - needs: - - build-sdk + needs: build-sdk steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" # Download the validation and api-v4 artifacts (built packages) - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation - - uses: actions/download-artifact@v3 + path: packages/validation/lib + - uses: actions/download-artifact@v4 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib # Create an api-v4 tarball - run: cd packages/api-v4 && npm pack --pack-destination ../../ @@ -162,37 +154,36 @@ jobs: test-manager: runs-on: ubuntu-latest - needs: - - build-sdk + needs: build-sdk steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation - - uses: actions/download-artifact@v3 + path: packages/validation/lib + - uses: actions/download-artifact@v4 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib - run: yarn --frozen-lockfile - run: yarn workspace linode-manager run test test-search: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules @@ -202,55 +193,53 @@ jobs: typecheck-manager: runs-on: ubuntu-latest - needs: - - build-sdk + needs: build-sdk steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation - - uses: actions/download-artifact@v3 + path: packages/validation/lib + - uses: actions/download-artifact@v4 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib - run: yarn --frozen-lockfile - run: yarn workspace linode-manager run typecheck build-manager: runs-on: ubuntu-latest if: github.ref == 'refs/heads/master' - needs: - - build-sdk + needs: build-sdk steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation + path: packages/validation/lib - uses: actions/download-artifact@v3 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib - run: yarn --frozen-lockfile - run: yarn workspace linode-manager run build - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: name: packages-manager-build path: packages/manager/build @@ -264,11 +253,11 @@ jobs: # If the validation publish failed we could have mismatched versions and a broken JS client - publish-validation steps: - - uses: actions/checkout@v3 - - uses: actions/download-artifact@v3 + - uses: actions/checkout@v4 + - uses: actions/download-artifact@v4 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib - uses: JS-DevTools/npm-publish@v1 id: npm-publish with: @@ -288,31 +277,30 @@ jobs: build-storybook: runs-on: ubuntu-latest - needs: - - build-sdk + needs: build-sdk env: NODE_OPTIONS: --max-old-space-size=4096 steps: - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules key: ${{ runner.os }}-${{ hashFiles('**/yarn.lock') }} - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: name: packages-validation-lib - path: packages/validation - - uses: actions/download-artifact@v3 + path: packages/validation/lib + - uses: actions/download-artifact@v4 with: name: packages-api-v4-lib - path: packages/api-v4 + path: packages/api-v4/lib - run: yarn --frozen-lockfile - run: yarn workspace linode-manager run build-storybook - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: name: storybook-build path: packages/manager/storybook-static @@ -320,11 +308,10 @@ jobs: publish-storybook: runs-on: ubuntu-latest if: github.ref == 'refs/heads/master' - needs: - - build-storybook + needs: build-storybook steps: - - uses: actions/checkout@v3 - - uses: actions/download-artifact@v3 + - uses: actions/checkout@v4 + - uses: actions/download-artifact@v4 with: name: storybook-build path: storybook/build diff --git a/.github/workflows/coverage.yml b/.github/workflows/coverage.yml index 84399b8323f..144f802594d 100644 --- a/.github/workflows/coverage.yml +++ b/.github/workflows/coverage.yml @@ -7,16 +7,16 @@ jobs: if: github.event.pull_request.draft == false runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: ref: ${{ github.base_ref }} # The base branch of the PR (develop) - name: Use Node.js v20.17 LTS - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules @@ -52,14 +52,14 @@ jobs: needs: base_branch steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - name: Use Node.js v20.17 LTS - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules @@ -88,13 +88,13 @@ jobs: echo "$pct" > current_code_coverage.txt - name: Upload PR Number Artifact - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: pr_number path: pr_number.txt - name: Upload Current Coverage Artifact - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: current_code_coverage path: current_code_coverage.txt diff --git a/.github/workflows/coverage_badge.yml b/.github/workflows/coverage_badge.yml index 0a73b196138..ca07bfd7f27 100644 --- a/.github/workflows/coverage_badge.yml +++ b/.github/workflows/coverage_badge.yml @@ -11,14 +11,14 @@ jobs: steps: - name: Checkout Code - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Use Node.js v20.17 LTS - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | **/node_modules diff --git a/.github/workflows/coverage_comment.yml b/.github/workflows/coverage_comment.yml index 07fe3682a9b..8ddd29c1ce6 100644 --- a/.github/workflows/coverage_comment.yml +++ b/.github/workflows/coverage_comment.yml @@ -15,10 +15,10 @@ jobs: steps: - name: Checkout Code - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Use Node.js v20.17 LTS - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: "20.17" diff --git a/.github/workflows/docs.yml b/.github/workflows/docs.yml index 0f73e3e58b6..4a81cf374f9 100644 --- a/.github/workflows/docs.yml +++ b/.github/workflows/docs.yml @@ -13,12 +13,12 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Setup Pages - uses: actions/configure-pages@v3 + uses: actions/configure-pages@v5 - - uses: oven-sh/setup-bun@v1 + - uses: oven-sh/setup-bun@v2 with: bun-version: 1.0.21 @@ -26,7 +26,7 @@ jobs: run: bunx vitepress@1.0.0-rc.35 build docs - name: Upload artifact - uses: actions/upload-pages-artifact@v2 + uses: actions/upload-pages-artifact@v3 with: path: docs/.vitepress/dist @@ -40,4 +40,4 @@ jobs: steps: - name: Deploy to GitHub Pages id: deployment - uses: actions/deploy-pages@v2 \ No newline at end of file + uses: actions/deploy-pages@v4 diff --git a/.github/workflows/e2e_schedule_and_push.yml b/.github/workflows/e2e_schedule_and_push.yml index 61176d8fb15..63578c1a25c 100644 --- a/.github/workflows/e2e_schedule_and_push.yml +++ b/.github/workflows/e2e_schedule_and_push.yml @@ -26,11 +26,11 @@ jobs: steps: - name: install command line utilities run: sudo apt-get install -y expect - - uses: actions/checkout@v3 - - uses: actions/setup-node@v3 + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 with: node-version: "20.17" - - uses: actions/cache@v3 + - uses: actions/cache@v4 with: path: | node_modules @@ -60,7 +60,7 @@ jobs: yarn build yarn start:manager:ci & - name: Run tests - uses: cypress-io/github-action@v5 + uses: cypress-io/github-action@v6 with: working-directory: packages/manager wait-on: "http://localhost:3000" diff --git a/.github/workflows/security_scan.yml b/.github/workflows/security_scan.yml index a7cb1b14fbe..37ed12a241f 100644 --- a/.github/workflows/security_scan.yml +++ b/.github/workflows/security_scan.yml @@ -12,7 +12,7 @@ jobs: container: image: returntocorp/semgrep steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 # Perform scanning using Semgrep # Pass even when it identifies issues or encounters errors. diff --git a/packages/manager/.changeset/pr-11009-tech-stories-1727310949305.md b/packages/manager/.changeset/pr-11009-tech-stories-1727310949305.md new file mode 100644 index 00000000000..4ad6da2dfc5 --- /dev/null +++ b/packages/manager/.changeset/pr-11009-tech-stories-1727310949305.md @@ -0,0 +1,5 @@ +--- +"@linode/manager": Tech Stories +--- + +Update Github Actions actions ([#11009](https://github.com/linode/manager/pull/11009)) From 4e25a942a09fef0e1ab56ea976ed8d6a8e39aef9 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 23 Oct 2024 11:37:28 -0500 Subject: [PATCH 04/45] Basci date picker component --- .../src/components/DatePicker/DatePicker.tsx | 81 +++++++++++++++++++ 1 file changed, 81 insertions(+) create mode 100644 packages/manager/src/components/DatePicker/DatePicker.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx new file mode 100644 index 00000000000..9a1aa35f26d --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -0,0 +1,81 @@ +import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; +import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; +import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; +import React, { useState } from 'react'; + +import { TextField } from 'src/components/TextField'; + +import type { DateTime } from 'luxon'; +import type { TextFieldProps } from 'src/components/TextField'; + +interface DatePickerProps { + /** If true, disables selecting future dates. Defaults to false. */ + disableFuture?: boolean; + /** If true, disables selecting past dates. Defaults to false. */ + disablePast?: boolean; + /** Error text to display below the input */ + errorText?: string; + /** Helper text to display below the input */ + helperText?: string; + /** Label to display for the date picker input */ + label?: string; + /** Callback function fired when the value changes */ + onChange?: (newDate: DateTime | null) => void; + /** Placeholder text for the date picker input */ + placeholder?: string; + /** Additional props to pass to the underlying TextField component */ + textFieldProps?: Omit; + /** The currently selected date */ + value?: DateTime | null; +} + +export const DatePicker = ({ + disableFuture = false, + disablePast = false, + errorText, + helperText, + label = 'Select a date', + onChange, + placeholder = 'Select a date', + textFieldProps, + value, + ...props +}: DatePickerProps) => { + const [selectedDate, setSelectedDate] = useState( + value || null + ); + + const onChangeHandler = (newDate: DateTime | null) => { + setSelectedDate(newDate); + if (onChange) { + onChange(newDate); + } + }; + + return ( + + ( + + ), + }, + }} + /> + + ); +}; From 423f4b7d1643c4d0a91ca4932f3ac40765d45f59 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Fri, 25 Oct 2024 14:19:53 -0500 Subject: [PATCH 05/45] Test coverage for date picker component --- .../components/DatePicker/DatePicker.test.tsx | 80 +++++++++++++++++++ .../src/components/DatePicker/DatePicker.tsx | 53 ++++++------ 2 files changed, 103 insertions(+), 30 deletions(-) create mode 100644 packages/manager/src/components/DatePicker/DatePicker.test.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.test.tsx b/packages/manager/src/components/DatePicker/DatePicker.test.tsx new file mode 100644 index 00000000000..e051d160ec5 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.test.tsx @@ -0,0 +1,80 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DatePicker } from './DatePicker'; + +import type { DatePickerProps } from './DatePicker'; + +const props: DatePickerProps = { + onChange: vi.fn(), + placeholder: 'Pick a date', + textFieldProps: { errorText: 'Invalid date', label: 'Select a date' }, + value: null, +}; + +describe('DatePicker', () => { + it('should render the DatePicker component', () => { + renderWithTheme(); + const DatePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + + expect(DatePickerField).toBeVisible(); + }); + + it('should handle value changes', async () => { + renderWithTheme(); + + const calendarButton = screen.getByRole('button', { name: 'Choose date' }); + + // Click the calendar button to open the date picker + await userEvent.click(calendarButton); + + // Find a date button to click (e.g., the 15th of the month) + const dateToSelect = screen.getByRole('gridcell', { name: '15' }); + await userEvent.click(dateToSelect); + + // Check if onChange was called after selecting a date + expect(props.onChange).toHaveBeenCalled(); + }); + + it('should display the error text when provided', () => { + renderWithTheme(); + const errorMessage = screen.getByText('Invalid date'); + expect(errorMessage).toBeVisible(); + }); + + it('should display the helper text when provided', () => { + renderWithTheme(); + const helperText = screen.getByText('Choose a valid date'); + expect(helperText).toBeVisible(); + }); + + it('should use the default format when no format is specified', () => { + renderWithTheme( + + ); + const datePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + expect(datePickerField).toHaveValue('2024-10-25'); + }); + + it('should handle the custom format correctly', () => { + renderWithTheme( + + ); + const datePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + expect(datePickerField).toHaveValue('25/10/2024'); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx index 9a1aa35f26d..89e26d7afff 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -1,20 +1,20 @@ import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; -import React, { useState } from 'react'; +import React from 'react'; import { TextField } from 'src/components/TextField'; +import type { DatePickerProps as MuiDatePickerProps } from '@mui/x-date-pickers/DatePicker'; import type { DateTime } from 'luxon'; import type { TextFieldProps } from 'src/components/TextField'; -interface DatePickerProps { - /** If true, disables selecting future dates. Defaults to false. */ - disableFuture?: boolean; - /** If true, disables selecting past dates. Defaults to false. */ - disablePast?: boolean; +export interface DatePickerProps + extends Omit, 'onChange' | 'value'> { /** Error text to display below the input */ errorText?: string; + /** Format of the date when rendered in the input field. */ + format?: string; /** Helper text to display below the input */ helperText?: string; /** Label to display for the date picker input */ @@ -30,23 +30,17 @@ interface DatePickerProps { } export const DatePicker = ({ - disableFuture = false, - disablePast = false, errorText, - helperText, + format = 'yyyy-MM-dd', + helperText = '', label = 'Select a date', onChange, - placeholder = 'Select a date', + placeholder = 'Pick a date', textFieldProps, - value, + value = null, ...props }: DatePickerProps) => { - const [selectedDate, setSelectedDate] = useState( - value || null - ); - const onChangeHandler = (newDate: DateTime | null) => { - setSelectedDate(newDate); if (onChange) { onChange(newDate); } @@ -55,26 +49,25 @@ export const DatePicker = ({ return ( ( - - ), + ...textFieldProps, + InputProps: { + ...textFieldProps?.InputProps, + }, + error: Boolean(errorText), + helperText, + label, + placeholder, }, }} + slots={{ + textField: TextField, // Use custom TextField + }} /> ); From f87fbce3cbbd325a357dfe466d56670644e6ae64 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 31 Oct 2024 09:54:24 -0500 Subject: [PATCH 06/45] DatePicker Stories --- .../DatePicker/DatePicker.stories.tsx | 43 +++++++++++++++++++ 1 file changed, 43 insertions(+) create mode 100644 packages/manager/src/components/DatePicker/DatePicker.stories.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.stories.tsx b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx new file mode 100644 index 00000000000..db3925b03eb --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx @@ -0,0 +1,43 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DatePicker } from './DatePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const ControlledExample: Story = { + render: () => { + const ControlledDatePicker = () => { + const [selectedDate, setSelectedDate] = React.useState( + DateTime.now() + ); + + const handleChange = (newDate: DateTime | null) => { + setSelectedDate(newDate); + action('Controlled date change')(newDate?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +const meta: Meta = { + component: DatePicker, + title: 'Components/DatePicker', +}; + +export default meta; From f47f91948da1fe48fc73890e90f9edfbf9232316 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 31 Oct 2024 10:00:05 -0500 Subject: [PATCH 07/45] Custom DateTimePicker component --- .../src/components/DatePicker/DatePicker.tsx | 4 +- .../DatePicker/DateTimePicker.test.tsx | 138 ++++++++++++ .../components/DatePicker/DateTimePicker.tsx | 199 ++++++++++++++++++ 3 files changed, 339 insertions(+), 2 deletions(-) create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.test.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx index 89e26d7afff..006bcb426d9 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -30,7 +30,6 @@ export interface DatePickerProps } export const DatePicker = ({ - errorText, format = 'yyyy-MM-dd', helperText = '', label = 'Select a date', @@ -59,14 +58,15 @@ export const DatePicker = ({ InputProps: { ...textFieldProps?.InputProps, }, - error: Boolean(errorText), helperText, label, placeholder, }, + ...props?.slotProps, }} slots={{ textField: TextField, // Use custom TextField + ...props?.slots, }} /> diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx new file mode 100644 index 00000000000..bc19b643ad3 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx @@ -0,0 +1,138 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DateTimePicker } from './DateTimePicker'; + +import type { DateTimePickerProps } from './DateTimePicker'; + +const defaultProps: DateTimePickerProps = { + label: 'Select Date and Time', + onApply: vi.fn(), + onCancel: vi.fn(), + onChange: vi.fn(), + placeholder: 'yyyy-MM-dd HH:mm', + value: DateTime.fromISO('2024-10-25T15:30:00'), +}; + +describe('DateTimePicker Component', () => { + it('should render the DateTimePicker component with the correct label and placeholder', () => { + renderWithTheme(); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + expect(textField).toBeVisible(); + expect(textField).toHaveAttribute('placeholder', 'yyyy-MM-dd HH:mm'); + }); + + it('should open the Popover when the TextField is clicked', async () => { + renderWithTheme(); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + await userEvent.click(textField); + expect(screen.getByRole('dialog')).toBeVisible(); // Verifying the Popover is open + }); + + it('should call onCancel when the Cancel button is clicked', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const cancelButton = screen.getByRole('button', { name: /Cancel/i }); + await userEvent.click(cancelButton); + expect(defaultProps.onCancel).toHaveBeenCalled(); + }); + + it('should call onApply when the Apply button is clicked', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const applyButton = screen.getByRole('button', { name: /Apply/i }); + await userEvent.click(applyButton); + expect(defaultProps.onApply).toHaveBeenCalled(); + expect(defaultProps.onChange).toHaveBeenCalledWith(expect.any(DateTime)); // Ensuring onChange was called with a DateTime object + }); + + it('should handle date changes correctly', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + + // Simulate selecting a date (e.g., 15th of the month) + const dateButton = screen.getByRole('gridcell', { name: '15' }); + await userEvent.click(dateButton); + + // Check that the displayed value has been updated correctly (this assumes the date format) + expect(defaultProps.onChange).toHaveBeenCalled(); + }); + + it.skip('should handle time changes correctly', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + + // Simulate changing time in the TimePicker (e.g., setting time to 10:45) + const timeInput = screen.getByLabelText(/Select Time/i); + await userEvent.type(timeInput, '10:45'); + expect(defaultProps.onChange).toHaveBeenCalled(); + }); + + it('should handle timezone changes correctly', async () => { + const timezoneChangeMock = vi.fn(); // Create a mock function + + const updatedProps = { + ...defaultProps, + timeZoneSelectProps: { onChange: timezoneChangeMock, value: 'UTC' }, + }; + + renderWithTheme(); + + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + + // Simulate selecting a timezone from the TimeZoneSelect + const timezoneInput = screen.getByPlaceholderText(/Choose a Timezone/i); + await userEvent.click(timezoneInput); + + // Select a timezone from the dropdown options + await userEvent.click( + screen.getByRole('option', { name: '(GMT -11:00) Niue Time' }) + ); + + // Click the Apply button to trigger the change + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Verify that the onChange function was called with the expected value + expect(timezoneChangeMock).toHaveBeenCalledWith('Pacific/Niue'); + }); + + it('should display the error text when provided', () => { + renderWithTheme( + + ); + expect(screen.getByText(/Invalid date-time/i)).toBeVisible(); + }); + + it('should format the date-time correctly when a custom format is provided', () => { + renderWithTheme( + + ); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + + expect(textField).toHaveValue('25/10/2024 15:30'); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx new file mode 100644 index 00000000000..a77c4eae60d --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -0,0 +1,199 @@ +import { Grid, Popover } from '@mui/material'; +import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; +import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; +import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; +import { TimePicker } from '@mui/x-date-pickers/TimePicker'; +import React, { useState } from 'react'; + +import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; +import { Box } from 'src/components/Box'; +import { Divider } from 'src/components/Divider'; +import { TextField } from 'src/components/TextField'; + +import { TimeZoneSelect } from './TimeZoneSelect'; + +import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; +import type { DateTime } from 'luxon'; + +export interface DateTimePickerProps { + /** Additional props for the DateCalendar */ + dateCalendarProps?: Partial>; + /** Error text for the date picker filed */ + errorText?: string; + /** Format for displaying the date-time */ + format?: string; + /** Label for the input field */ + label?: string; + /** Callback when the "Apply" button is clicked */ + onApply?: () => void; + /** Callback when the "Cancel" button is clicked */ + onCancel?: () => void; + /** Callback when date-time changes */ + onChange?: (dateTime: DateTime | null) => void; + /** Placeholder text for the input field */ + placeholder?: string; + timeSelectProps?: { + label?: string; + onChange?: (time: string) => void; + value?: null | string; + }; + /** Props for customizing the TimeZoneSelect component */ + timeZoneSelectProps?: { + label?: string; + onChange?: (timezone: string) => void; + value?: null | string; + }; + /** Initial or controlled dateTime value */ + value?: DateTime | null; +} + +export const DateTimePicker = ({ + dateCalendarProps = {}, + errorText = '', + format = 'yyyy-MM-dd HH:mm', + label = 'Select Date and Time', + onApply, + onCancel, + onChange, + placeholder = 'yyyy-MM-dd HH:mm', + timeSelectProps, + timeZoneSelectProps, + value = null, +}: DateTimePickerProps) => { + const [anchorEl, setAnchorEl] = useState(null); + const [dateTime, setDateTime] = useState(value); + const [timezone, setTimezone] = useState( + timeZoneSelectProps?.value || null + ); + + const handleOpen = (event: React.MouseEvent) => { + setAnchorEl(event.currentTarget); + }; + + const handleClose = () => { + setAnchorEl(null); + if (onCancel) { + onCancel(); + } + }; + + const handleApply = () => { + setAnchorEl(null); + if (onChange) { + onChange(dateTime); + } + if (onApply) { + onApply(); + } + }; + + const handleDateChange = (newDate: DateTime | null) => { + setDateTime((prev) => + newDate + ? newDate.set({ + hour: prev?.hour || 0, + minute: prev?.minute || 0, + }) + : null + ); + }; + + const handleTimeChange = (newTime: DateTime | null) => { + if (newTime) { + setDateTime((prev) => { + if (prev) { + // Ensure hour and minute are valid numbers + const newHour = newTime.hour; + const newMinute = newTime.minute; + + if (typeof newHour === 'number' && typeof newMinute === 'number') { + return prev.set({ hour: newHour, minute: newMinute }); + } + } + // Return the current `prev` value if newTime is invalid + return prev; + }); + } + }; + + const handleTimezoneChange = (newTimezone: string) => { + setTimezone(newTimezone); + setDateTime((prev) => (prev ? prev.setZone(newTimezone) : null)); + if (timeZoneSelectProps?.onChange) { + timeZoneSelectProps.onChange(newTimezone); + } + }; + + return ( + + + + + + + + + + + + + + + + + + + ({ + marginBottom: theme.spacing(1), + marginRight: theme.spacing(2), + })} + /> + + + + ); +}; From 2578b837a0656cee6829eb95d2d2718643a21e11 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 31 Oct 2024 10:01:06 -0500 Subject: [PATCH 08/45] Reusable TimeZone Select Component --- .../components/DatePicker/TimeZoneSelect.tsx | 63 +++++++++++++++++++ 1 file changed, 63 insertions(+) create mode 100644 packages/manager/src/components/DatePicker/TimeZoneSelect.tsx diff --git a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx new file mode 100644 index 00000000000..3ec726b6ef6 --- /dev/null +++ b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx @@ -0,0 +1,63 @@ +import { DateTime } from 'luxon'; +import React from 'react'; + +import { timezones } from 'src/assets/timezones/timezones'; +import { Autocomplete } from 'src/components/Autocomplete/Autocomplete'; + +type Timezone = typeof timezones[number]; + +interface TimeZoneSelectProps { + disabled?: boolean; + errorText?: string; + label?: string; + onChange: (timezone: string) => void; + value: null | string; +} + +const getOptionLabel = ({ label, offset }: Timezone) => { + const minutes = (Math.abs(offset) % 60).toLocaleString(undefined, { + minimumIntegerDigits: 2, + useGrouping: false, + }); + const hours = Math.floor(Math.abs(offset) / 60); + const isPositive = Math.abs(offset) === offset ? '+' : '-'; + + return `(GMT ${isPositive}${hours}:${minutes}) ${label}`; +}; + +const getTimezoneOptions = () => { + return timezones + .map((tz) => { + const offset = DateTime.now().setZone(tz.name).offset; + const label = getOptionLabel({ ...tz, offset }); + return { label, offset, value: tz.name }; + }) + .sort((a, b) => a.offset - b.offset); +}; + +const timezoneOptions = getTimezoneOptions(); + +export const TimeZoneSelect = ({ + disabled = false, + errorText, + label = 'Timezone', + onChange, + value, +}: TimeZoneSelectProps) => { + return ( + option.value === value) ?? undefined + } + autoHighlight + disableClearable + disabled={disabled} + errorText={errorText} + fullWidth + label={label} + onChange={(e, option) => onChange(option?.value || '')} + options={timezoneOptions} + placeholder="Choose a Timezone" + /> + ); +}; From 43810e75d75ec4417799ce779e8620de0ce16f5d Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 31 Oct 2024 10:18:17 -0500 Subject: [PATCH 09/45] Create custom DateTimeRangePicker component --- .../DateTimeRangePicker.stories.tsx | 77 ++++++++++++ .../DatePicker/DateTimeRangePicker.test.tsx | 112 ++++++++++++++++++ .../DatePicker/DateTimeRangePicker.tsx | 89 ++++++++++++++ 3 files changed, 278 insertions(+) create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx new file mode 100644 index 00000000000..70ad6b2a39d --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx @@ -0,0 +1,77 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DateTimeRangePicker } from './DateTimeRangePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const Default: Story = { + args: { + endLabel: 'End Date and Time', + format: 'yyyy-MM-dd HH:mm', + onChange: action('DateTime range changed'), + startLabel: 'Start Date and Time', + }, + render: (args) => , +}; + +export const WithInitialValues: Story = { + render: () => { + const ComponentWithState: React.FC = () => { + const [start, setStart] = React.useState( + DateTime.now().minus({ days: 1 }) + ); + const [end, setEnd] = React.useState(DateTime.now()); + + const handleDateChange = ( + newStart: DateTime | null, + newEnd: DateTime | null + ) => { + setStart(newStart); + setEnd(newEnd); + action('DateTime range changed')(newStart?.toISO(), newEnd?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +const meta: Meta = { + argTypes: { + endLabel: { + control: 'text', + description: 'Label for the end date picker', + }, + format: { + control: 'text', + description: 'Format for displaying the date-time', + }, + onChange: { + action: 'DateTime range changed', + description: 'Callback when the date-time range changes', + }, + startLabel: { + control: 'text', + description: 'Label for the start date picker', + }, + }, + component: DateTimeRangePicker, + title: 'Components/DateTimeRangePicker', +}; + +export default meta; diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx new file mode 100644 index 00000000000..843678ac4b9 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -0,0 +1,112 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DateTimeRangePicker } from './DateTimeRangePicker'; + +describe('DateTimeRangePicker Component', () => { + const onChangeMock = vi.fn(); + + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should render start and end DateTimePickers with correct labels', () => { + renderWithTheme(); + + expect(screen.getByLabelText('Start Date')).toBeVisible(); + expect(screen.getByLabelText('End Date and Time')).toBeVisible(); + }); + + it('should call onChange when start date is changed', async () => { + renderWithTheme(); + + // Open start date picker + await userEvent.click(screen.getByLabelText('Start Date')); + + await userEvent.click(screen.getByRole('gridcell', { name: '10' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Check if the onChange function is called with the expected DateTime value + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ day: 10 }), + null + ); + }); + + it.skip('should call onChange when end date is changed', async () => { + renderWithTheme( + + ); + + // Open the end date picker + await userEvent.click( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ); + + // Confirm the dialog is visible and wait for it + expect(screen.getByRole('dialog')).toBeVisible(); + + // Simulate selecting a date (e.g., 15th of the month) + const element = screen.getByRole('gridcell', { name: '2' }); + + await userEvent.click(element); + // await userEvent.click(screen.getByRole('gridcell', { name: '1' })); + + // Click the 'Apply' button and wait for onChange to be called + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Verify onChange is called with the appropriate arguments + expect(onChangeMock).toHaveBeenCalledWith(null, expect.any(DateTime)); + }); + + it.skip('should show error when end date-time is before start date-time', async () => { + renderWithTheme(); + + // Set start date-time to the 15th + const startDateField = screen.getByLabelText('Start Date'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Set end date-time to the 10th (before start date-time) + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '10' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + expect(screen.getAllByText('Invalid dae and Time')).toHaveLength(2); + }); + + it.skip('should clear the error when a valid end date-time is selected', async () => { + renderWithTheme(); + + // Set start date-time to the 10th + const startDateField = screen.getByLabelText('Start Date'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '10' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Set an invalid end date-time (e.g., 5th) + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '5' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + expect(screen.getAllByText('Invalid dae and Time')).toHaveLength(2); + + // Select a valid end date-time (e.g., 15th) + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + expect(screen.queryByText('Invalid dae and Time')).not.toBeInTheDocument(); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx new file mode 100644 index 00000000000..95d9e12a45e --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -0,0 +1,89 @@ +import { DateTime } from 'luxon'; +import React, { useState } from 'react'; + +import { Box } from 'src/components/Box'; + +import { DateTimePicker } from './DateTimePicker'; + +interface DateTimeRangePickerProps { + /** Initial or controlled value for the end date-time */ + endDateTimeValue?: DateTime | null; + /** Custom labels for the start and end date/time fields */ + endLabel?: string; + /** Format for displaying the date-time */ + format?: string; + /** Callback when the date-time range changes */ + onChange?: (start: DateTime | null, end: DateTime | null) => void; + /** Initial or controlled value for the start date-time */ + startDateTimeValue?: DateTime | null; + /** Custom labels for the start and end date/time fields */ + startLabel?: string; +} + +export const DateTimeRangePicker = ({ + endDateTimeValue = null, + endLabel = 'End Date and Time', + format = 'yyyy-MM-dd HH:mm', + onChange, + startDateTimeValue = null, + startLabel = 'Start Date', +}: DateTimeRangePickerProps) => { + const [startDateTime, setStartDateTime] = useState( + startDateTimeValue + ); + const [endDateTime, setEndDateTime] = useState( + endDateTimeValue + ); + const [error, setError] = useState(); + + const handleStartDateTimeChange = (newStart: DateTime | null) => { + setStartDateTime(newStart); + + // Reset error if the selection is valid + if (endDateTime && newStart && endDateTime >= newStart) { + setError(undefined); + } + + if (onChange) { + onChange(newStart, endDateTime); + } + }; + + const handleEndDateTimeChange = (newEnd: DateTime | null) => { + // Check if the end date is before the start date + if (startDateTime && newEnd && newEnd < startDateTime) { + setError('Invalid dae and Time'); + } else { + setEndDateTime(newEnd); + setError(undefined); // Clear the error if the selection is valid + } + + if (onChange) { + onChange(startDateTime, newEnd); + } + }; + + return ( + + + + + ); +}; From 3b6f981363089f989663e16e3e4cfd97b420cc18 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 5 Nov 2024 19:45:30 -0600 Subject: [PATCH 10/45] Storybook for DateTimePicker --- .../DatePicker/DateTimePicker.stories.tsx | 93 +++++++++++++++++++ .../components/DatePicker/DateTimePicker.tsx | 2 +- .../DatePicker/DateTimeRangePicker.tsx | 3 +- 3 files changed, 95 insertions(+), 3 deletions(-) create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx new file mode 100644 index 00000000000..32d65dfa0b7 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx @@ -0,0 +1,93 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DateTimePicker } from './DateTimePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const ControlledExample: Story = { + render: () => { + const ControlledDateTimePicker = () => { + const [ + selectedDateTime, + setSelectedDateTime, + ] = React.useState(DateTime.now()); + + const handleChange = (newDateTime: DateTime | null) => { + setSelectedDateTime(newDateTime); + action('Controlled dateTime change')(newDateTime?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +export const DefaultExample: Story = { + args: { + label: 'Default Date-Time Picker', + onApply: action('Apply clicked'), + onCancel: action('Cancel clicked'), + onChange: action('Date-Time selected'), + placeholder: 'yyyy-MM-dd HH:mm', + }, +}; + +export const WithErrorText: Story = { + args: { + errorText: 'This field is required', + label: 'Date-Time Picker with Error', + onApply: action('Apply clicked with error'), + onCancel: action('Cancel clicked with error'), + onChange: action('Date-Time selected with error'), + placeholder: 'yyyy-MM-dd HH:mm', + }, +}; + +const meta: Meta = { + argTypes: { + errorText: { + control: { type: 'text' }, + }, + format: { + control: { type: 'text' }, + }, + label: { + control: { type: 'text' }, + }, + onApply: { action: 'applyClicked' }, + onCancel: { action: 'cancelClicked' }, + onChange: { action: 'dateTimeChanged' }, + placeholder: { + control: { type: 'text' }, + }, + }, + args: { + format: 'yyyy-MM-dd HH:mm', + label: 'Date-Time Picker', + placeholder: 'Select a date and time', + }, + component: DateTimePicker, + title: 'Components/DateTimePicker', +}; + +export default meta; diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index a77c4eae60d..2fac07dfaf3 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,3 +1,4 @@ +import { Box } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; @@ -6,7 +7,6 @@ import { TimePicker } from '@mui/x-date-pickers/TimePicker'; import React, { useState } from 'react'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; -import { Box } from 'src/components/Box'; import { Divider } from 'src/components/Divider'; import { TextField } from 'src/components/TextField'; diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 95d9e12a45e..f7a658178b3 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,8 +1,7 @@ +import { Box } from '@linode/ui'; import { DateTime } from 'luxon'; import React, { useState } from 'react'; -import { Box } from 'src/components/Box'; - import { DateTimePicker } from './DateTimePicker'; interface DateTimeRangePickerProps { From 9b10623bdcb63b88c3c9c00ca56e9bfff503d492 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 7 Nov 2024 12:20:29 -0600 Subject: [PATCH 11/45] Fix tests and remove console warnings --- .../components/DatePicker/DateTimePicker.tsx | 2 +- .../DatePicker/DateTimeRangePicker.test.tsx | 67 ++----------------- 2 files changed, 8 insertions(+), 61 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 2fac07dfaf3..1d7e05b8237 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -133,7 +133,7 @@ export const DateTimePicker = ({ ? `${dateTime.toFormat(format)}${ timezone ? ` (${timezone})` : '' }` - : null + : '' } InputProps={{ readOnly: true }} errorText={errorText} diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index 843678ac4b9..f3421b684c3 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -1,6 +1,5 @@ import { screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; -import { DateTime } from 'luxon'; import * as React from 'react'; import { renderWithTheme } from 'src/utilities/testHelpers'; @@ -37,37 +36,7 @@ describe('DateTimeRangePicker Component', () => { ); }); - it.skip('should call onChange when end date is changed', async () => { - renderWithTheme( - - ); - - // Open the end date picker - await userEvent.click( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ); - - // Confirm the dialog is visible and wait for it - expect(screen.getByRole('dialog')).toBeVisible(); - - // Simulate selecting a date (e.g., 15th of the month) - const element = screen.getByRole('gridcell', { name: '2' }); - - await userEvent.click(element); - // await userEvent.click(screen.getByRole('gridcell', { name: '1' })); - - // Click the 'Apply' button and wait for onChange to be called - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - // Verify onChange is called with the appropriate arguments - expect(onChangeMock).toHaveBeenCalledWith(null, expect.any(DateTime)); - }); - - it.skip('should show error when end date-time is before start date-time', async () => { + it('should show error when end date-time is before start date-time', async () => { renderWithTheme(); // Set start date-time to the 15th @@ -76,37 +45,15 @@ describe('DateTimeRangePicker Component', () => { await userEvent.click(screen.getByRole('gridcell', { name: '15' })); await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - // Set end date-time to the 10th (before start date-time) - const endDateField = screen.getByLabelText('End Date and Time'); - await userEvent.click(endDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '10' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - expect(screen.getAllByText('Invalid dae and Time')).toHaveLength(2); - }); - - it.skip('should clear the error when a valid end date-time is selected', async () => { - renderWithTheme(); - - // Set start date-time to the 10th - const startDateField = screen.getByLabelText('Start Date'); - await userEvent.click(startDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '10' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - // Set an invalid end date-time (e.g., 5th) + // Open the end date picker const endDateField = screen.getByLabelText('End Date and Time'); await userEvent.click(endDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '5' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - expect(screen.getAllByText('Invalid dae and Time')).toHaveLength(2); - - // Select a valid end date-time (e.g., 15th) - await userEvent.click(endDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '15' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Check if the date before the start date is disabled via a class or attribute + const invalidDate = screen.getByRole('gridcell', { name: '10' }); + expect(invalidDate).toHaveClass('Mui-disabled'); // or check for the specific attribute used - expect(screen.queryByText('Invalid dae and Time')).not.toBeInTheDocument(); + // Confirm error message is not shown since the click was blocked + expect(screen.queryByText('Invalid date and Time')).not.toBeInTheDocument(); }); }); From 11d878004ebca1ebea35a703bb4d6a96797e3a4d Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 18 Nov 2024 09:38:27 -0600 Subject: [PATCH 12/45] changeset --- packages/manager/.changeset/pr-11151-added-1731944151381.md | 5 +++++ .../manager/src/components/DatePicker/DateTimePicker.tsx | 2 +- .../features/GlobalNotifications/CreditCardExpiredBanner.tsx | 2 +- 3 files changed, 7 insertions(+), 2 deletions(-) create mode 100644 packages/manager/.changeset/pr-11151-added-1731944151381.md diff --git a/packages/manager/.changeset/pr-11151-added-1731944151381.md b/packages/manager/.changeset/pr-11151-added-1731944151381.md new file mode 100644 index 00000000000..1a64466eea4 --- /dev/null +++ b/packages/manager/.changeset/pr-11151-added-1731944151381.md @@ -0,0 +1,5 @@ +--- +"@linode/manager": Added +--- + +New DatePicker Component ([#11151](https://github.com/linode/manager/pull/11151)) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 1d7e05b8237..2fca86ab17f 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,3 +1,4 @@ +import { Divider } from '@linode/ui'; import { Box } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; @@ -7,7 +8,6 @@ import { TimePicker } from '@mui/x-date-pickers/TimePicker'; import React, { useState } from 'react'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; -import { Divider } from 'src/components/Divider'; import { TextField } from 'src/components/TextField'; import { TimeZoneSelect } from './TimeZoneSelect'; diff --git a/packages/manager/src/features/GlobalNotifications/CreditCardExpiredBanner.tsx b/packages/manager/src/features/GlobalNotifications/CreditCardExpiredBanner.tsx index dbbe22436e8..c386feb9b4c 100644 --- a/packages/manager/src/features/GlobalNotifications/CreditCardExpiredBanner.tsx +++ b/packages/manager/src/features/GlobalNotifications/CreditCardExpiredBanner.tsx @@ -1,8 +1,8 @@ +import { Button } from '@linode/ui'; import { Typography } from '@mui/material'; import * as React from 'react'; import { useHistory } from 'react-router-dom'; -import { Button } from 'src/components/Button/Button'; import { DismissibleBanner } from 'src/components/DismissibleBanner/DismissibleBanner'; import { useAccount } from 'src/queries/account/account'; import { isCreditCardExpired } from 'src/utilities/creditCard'; From 7aec8c958694af09bdf801c1d97f2d75bfad4127 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 20 Nov 2024 05:17:53 -0600 Subject: [PATCH 13/45] Update packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx Co-authored-by: Connie Liu <139280159+coliu-akamai@users.noreply.github.com> --- .../manager/src/components/DatePicker/DateTimeRangePicker.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index f7a658178b3..ed57af7c37c 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -51,7 +51,7 @@ export const DateTimeRangePicker = ({ const handleEndDateTimeChange = (newEnd: DateTime | null) => { // Check if the end date is before the start date if (startDateTime && newEnd && newEnd < startDateTime) { - setError('Invalid dae and Time'); + setError('Invalid date and time'); } else { setEndDateTime(newEnd); setError(undefined); // Clear the error if the selection is valid From 0c379a2f692ba5256aef483a5152845b08d467b1 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Thu, 21 Nov 2024 13:02:31 -0600 Subject: [PATCH 14/45] Adjust styles for DatePicker --- .../src/components/DatePicker/DatePicker.tsx | 31 +++++++++++++++---- .../DatePicker/DateTimePicker.test.tsx | 12 ------- 2 files changed, 25 insertions(+), 18 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx index 006bcb426d9..20573d41f7f 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -1,3 +1,4 @@ +import { useTheme } from '@mui/material/styles'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; @@ -39,6 +40,8 @@ export const DatePicker = ({ value = null, ...props }: DatePickerProps) => { + const theme = useTheme(); + const onChangeHandler = (newDate: DateTime | null) => { if (onChange) { onChange(newDate); @@ -50,23 +53,39 @@ export const DatePicker = ({ diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx index bc19b643ad3..ca9bc882340 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx @@ -72,18 +72,6 @@ describe('DateTimePicker Component', () => { expect(defaultProps.onChange).toHaveBeenCalled(); }); - it.skip('should handle time changes correctly', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - - // Simulate changing time in the TimePicker (e.g., setting time to 10:45) - const timeInput = screen.getByLabelText(/Select Time/i); - await userEvent.type(timeInput, '10:45'); - expect(defaultProps.onChange).toHaveBeenCalled(); - }); - it('should handle timezone changes correctly', async () => { const timezoneChangeMock = vi.fn(); // Create a mock function From 194b0d2a4526ec4b42edf3fefd54e3b5195365a8 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Fri, 22 Nov 2024 06:51:32 -0600 Subject: [PATCH 15/45] Adjust styles for DateTimePicker --- .../components/DatePicker/DateTimePicker.tsx | 46 +++++++++++++++++-- .../components/DatePicker/TimeZoneSelect.tsx | 5 +- 2 files changed, 45 insertions(+), 6 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 2fca86ab17f..2f01eabb20a 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,5 +1,5 @@ -import { Divider } from '@linode/ui'; import { Box } from '@linode/ui'; +import { Divider } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; @@ -14,6 +14,7 @@ import { TimeZoneSelect } from './TimeZoneSelect'; import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; import type { DateTime } from 'luxon'; +import type { TextFieldProps } from 'src/components/TextField'; export interface DateTimePickerProps { /** Additional props for the DateCalendar */ @@ -66,6 +67,11 @@ export const DateTimePicker = ({ timeZoneSelectProps?.value || null ); + const textFieldProps: TextFieldProps = { + label: timeSelectProps?.label ?? 'Select Time', + noMarginTop: true, + }; + const handleOpen = (event: React.MouseEvent) => { setAnchorEl(event.currentTarget); }; @@ -151,24 +157,56 @@ export const DateTimePicker = ({ > ({ + '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { + justifyContent: 'space-between', + }, + '& .MuiDayCalendar-weekDayLabel': { + fontSize: '0.875rem', + }, + '& .MuiPickersCalendarHeader-label': { + fontFamily: theme.font.bold, + }, + '& .MuiPickersCalendarHeader-root': { + borderBottom: `1px solid ${theme.borderColors.divider}`, + fontSize: '0.875rem', + paddingBottom: theme.spacing(1), + }, + '& .MuiPickersDay-root': { + fontSize: '0.875rem', + margin: `${theme.spacing(0.5)}px`, + }, + borderRadius: `${theme.spacing(2)}`, + borderWidth: '0px', + })} onChange={handleDateChange} value={dateTime} {...dateCalendarProps} /> - + - + diff --git a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx index 3ec726b6ef6..19312098007 100644 --- a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx +++ b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx @@ -10,6 +10,7 @@ interface TimeZoneSelectProps { disabled?: boolean; errorText?: string; label?: string; + noMarginTop?: boolean; onChange: (timezone: string) => void; value: null | string; } @@ -41,6 +42,7 @@ export const TimeZoneSelect = ({ disabled = false, errorText, label = 'Timezone', + noMarginTop = false, onChange, value, }: TimeZoneSelectProps) => { @@ -50,11 +52,10 @@ export const TimeZoneSelect = ({ timezoneOptions.find((option) => option.value === value) ?? undefined } autoHighlight - disableClearable disabled={disabled} errorText={errorText} - fullWidth label={label} + noMarginTop={noMarginTop} onChange={(e, option) => onChange(option?.value || '')} options={timezoneOptions} placeholder="Choose a Timezone" From b2d591f58823c508e76ff21e6d0cf7dd88190916 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 2 Dec 2024 11:21:40 -0600 Subject: [PATCH 16/45] update imports --- packages/manager/src/components/DatePicker/DatePicker.tsx | 5 ++--- .../manager/src/components/DatePicker/DateTimePicker.tsx | 4 ++-- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx index 20573d41f7f..a3a5a722e82 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -1,14 +1,13 @@ +import { TextField } from '@linode/ui'; import { useTheme } from '@mui/material/styles'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; import React from 'react'; -import { TextField } from 'src/components/TextField'; - +import type { TextFieldProps } from '@linode/ui'; import type { DatePickerProps as MuiDatePickerProps } from '@mui/x-date-pickers/DatePicker'; import type { DateTime } from 'luxon'; -import type { TextFieldProps } from 'src/components/TextField'; export interface DatePickerProps extends Omit, 'onChange' | 'value'> { diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 2f01eabb20a..bc37175a636 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,3 +1,4 @@ +import { TextField } from '@linode/ui'; import { Box } from '@linode/ui'; import { Divider } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; @@ -8,13 +9,12 @@ import { TimePicker } from '@mui/x-date-pickers/TimePicker'; import React, { useState } from 'react'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; -import { TextField } from 'src/components/TextField'; import { TimeZoneSelect } from './TimeZoneSelect'; +import type { TextFieldProps } from '@linode/ui'; import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; import type { DateTime } from 'luxon'; -import type { TextFieldProps } from 'src/components/TextField'; export interface DateTimePickerProps { /** Additional props for the DateCalendar */ From efcfe5ce5a926a9b12fbc7c5eb6c3379122e3366 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 2 Dec 2024 13:35:44 -0600 Subject: [PATCH 17/45] Render time and timezone conditionally in DateTimePicker component --- .../DatePicker/DateTimePicker.test.tsx | 17 ++++++ .../components/DatePicker/DateTimePicker.tsx | 55 +++++++++++-------- 2 files changed, 50 insertions(+), 22 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx index ca9bc882340..12f1795a747 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx @@ -123,4 +123,21 @@ describe('DateTimePicker Component', () => { expect(textField).toHaveValue('25/10/2024 15:30'); }); + it('should not render the time selector when showTime is false', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const timePicker = screen.queryByLabelText(/Select Time/i); // Label from timeSelectProps + expect(timePicker).not.toBeInTheDocument(); + }); + + it('should not render the timezone selector when showTimeZone is false', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const timeZoneSelect = screen.queryByLabelText(/Timezone/i); // Label from timeZoneSelectProps + expect(timeZoneSelect).not.toBeInTheDocument(); + }); }); diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index bc37175a636..c6ac1834166 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -33,6 +33,11 @@ export interface DateTimePickerProps { onChange?: (dateTime: DateTime | null) => void; /** Placeholder text for the input field */ placeholder?: string; + /** Whether to show the time selector */ + showTime?: boolean; + /** Whether to show the timezone selector */ + showTimeZone?: boolean; + /** Props for customizing the TimePicker component */ timeSelectProps?: { label?: string; onChange?: (time: string) => void; @@ -57,6 +62,8 @@ export const DateTimePicker = ({ onCancel, onChange, placeholder = 'yyyy-MM-dd HH:mm', + showTime = true, + showTimeZone = true, timeSelectProps, timeZoneSelectProps, value = null, @@ -188,29 +195,33 @@ export const DateTimePicker = ({ spacing={2} sx={{ display: 'flex', justifyContent: 'space-between' }} > - - + - - - - + textField: textFieldProps, + }} + onChange={handleTimeChange} + slots={{ textField: TextField }} + value={dateTime} + /> + + )} + {showTimeZone && ( + + + + )} From 2ca7adf92d8b9c40d9c7b0d751e8dbea45b3832d Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 3 Dec 2024 10:46:16 -0600 Subject: [PATCH 18/45] Move DatePicker to UI package --- .../src/components/DatePicker/DatePicker.stories.tsx | 0 .../src/components/DatePicker/DatePicker.test.tsx | 0 .../{manager => ui}/src/components/DatePicker/DatePicker.tsx | 0 .../src/components/DatePicker/DateTimePicker.stories.tsx | 0 .../src/components/DatePicker/DateTimePicker.test.tsx | 0 .../{manager => ui}/src/components/DatePicker/DateTimePicker.tsx | 0 .../src/components/DatePicker/DateTimeRangePicker.stories.tsx | 0 .../src/components/DatePicker/DateTimeRangePicker.test.tsx | 0 .../src/components/DatePicker/DateTimeRangePicker.tsx | 0 .../{manager => ui}/src/components/DatePicker/TimeZoneSelect.tsx | 0 packages/ui/src/components/DatePicker/index.ts | 1 + packages/ui/src/components/index.ts | 1 + 12 files changed, 2 insertions(+) rename packages/{manager => ui}/src/components/DatePicker/DatePicker.stories.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DatePicker.test.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DatePicker.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimePicker.stories.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimePicker.test.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimePicker.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimeRangePicker.stories.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimeRangePicker.test.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/DateTimeRangePicker.tsx (100%) rename packages/{manager => ui}/src/components/DatePicker/TimeZoneSelect.tsx (100%) create mode 100644 packages/ui/src/components/DatePicker/index.ts diff --git a/packages/manager/src/components/DatePicker/DatePicker.stories.tsx b/packages/ui/src/components/DatePicker/DatePicker.stories.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DatePicker.stories.tsx rename to packages/ui/src/components/DatePicker/DatePicker.stories.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.test.tsx b/packages/ui/src/components/DatePicker/DatePicker.test.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DatePicker.test.tsx rename to packages/ui/src/components/DatePicker/DatePicker.test.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/ui/src/components/DatePicker/DatePicker.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DatePicker.tsx rename to packages/ui/src/components/DatePicker/DatePicker.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx rename to packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.test.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimePicker.test.tsx rename to packages/ui/src/components/DatePicker/DateTimePicker.test.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimePicker.tsx rename to packages/ui/src/components/DatePicker/DateTimePicker.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx rename to packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx rename to packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx rename to packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx diff --git a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx b/packages/ui/src/components/DatePicker/TimeZoneSelect.tsx similarity index 100% rename from packages/manager/src/components/DatePicker/TimeZoneSelect.tsx rename to packages/ui/src/components/DatePicker/TimeZoneSelect.tsx diff --git a/packages/ui/src/components/DatePicker/index.ts b/packages/ui/src/components/DatePicker/index.ts new file mode 100644 index 00000000000..a48b62e4dfc --- /dev/null +++ b/packages/ui/src/components/DatePicker/index.ts @@ -0,0 +1 @@ +export * from './DatePicker'; diff --git a/packages/ui/src/components/index.ts b/packages/ui/src/components/index.ts index db690f6940e..f64e7b4105a 100644 --- a/packages/ui/src/components/index.ts +++ b/packages/ui/src/components/index.ts @@ -26,3 +26,4 @@ export * from './Tooltip'; export * from './TooltipIcon'; export * from './Typography'; export * from './VisibilityTooltip'; +export * from './DatePicker'; From 31dc03ea9c5625f56cd0496721c84d8d10969a37 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 3 Dec 2024 14:43:53 -0600 Subject: [PATCH 19/45] Add DatePicker dependencies --- packages/ui/package.json | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/packages/ui/package.json b/packages/ui/package.json index 342fef3d1ba..3a85d11a12a 100644 --- a/packages/ui/package.json +++ b/packages/ui/package.json @@ -20,7 +20,9 @@ "@emotion/styled": "^11.11.0", "@mui/material": "^5.14.7", "react": "^18.2.0", - "react-dom": "^18.2.0" + "react-dom": "^18.2.0", + "@mui/x-date-pickers": "^7.12.0", + "luxon": "3.4.4" }, "scripts": { "start": "tsc -w --preserveWatchOutput", @@ -57,4 +59,4 @@ "prettier": "~2.2.1", "vite-plugin-svgr": "^3.2.0" } -} +} \ No newline at end of file From cd875be5b91f4337892e2af4091731fb38f80f77 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 3 Dec 2024 18:15:21 -0600 Subject: [PATCH 20/45] Code cleanup --- .../components/DatePicker/DateTimePicker.tsx | 75 ++++++++++++------- 1 file changed, 47 insertions(+), 28 deletions(-) diff --git a/packages/ui/src/components/DatePicker/DateTimePicker.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.tsx index c6ac1834166..4d65dc6f1b9 100644 --- a/packages/ui/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/ui/src/components/DatePicker/DateTimePicker.tsx @@ -1,25 +1,26 @@ import { TextField } from '@linode/ui'; -import { Box } from '@linode/ui'; import { Divider } from '@linode/ui'; +import { Box } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; import { TimePicker } from '@mui/x-date-pickers/TimePicker'; -import React, { useState } from 'react'; +import React, { useEffect, useState } from 'react'; import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; import { TimeZoneSelect } from './TimeZoneSelect'; import type { TextFieldProps } from '@linode/ui'; +import type { Theme } from '@mui/material/styles'; import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; import type { DateTime } from 'luxon'; export interface DateTimePickerProps { /** Additional props for the DateCalendar */ dateCalendarProps?: Partial>; - /** Error text for the date picker filed */ + /** Error text for the date picker field */ errorText?: string; /** Format for displaying the date-time */ format?: string; @@ -64,21 +65,35 @@ export const DateTimePicker = ({ placeholder = 'yyyy-MM-dd HH:mm', showTime = true, showTimeZone = true, - timeSelectProps, - timeZoneSelectProps, + timeSelectProps = {}, + timeZoneSelectProps = {}, value = null, }: DateTimePickerProps) => { const [anchorEl, setAnchorEl] = useState(null); - const [dateTime, setDateTime] = useState(value); - const [timezone, setTimezone] = useState( - timeZoneSelectProps?.value || null + const [selectedDateTime, setSelectedDateTime] = useState( + value + ); + const [selectedTimeZone, setSelectedTimeZone] = useState( + timeZoneSelectProps.value || null ); - const textFieldProps: TextFieldProps = { + const TimePickerFieldProps: TextFieldProps = { label: timeSelectProps?.label ?? 'Select Time', noMarginTop: true, }; + useEffect(() => { + if (value) { + setSelectedDateTime(value); + } + }, [value]); + + useEffect(() => { + if (timeZoneSelectProps.value) { + setSelectedTimeZone(timeZoneSelectProps.value); + } + }, [timeZoneSelectProps.value]); + const handleOpen = (event: React.MouseEvent) => { setAnchorEl(event.currentTarget); }; @@ -93,7 +108,7 @@ export const DateTimePicker = ({ const handleApply = () => { setAnchorEl(null); if (onChange) { - onChange(dateTime); + onChange(selectedDateTime); } if (onApply) { onApply(); @@ -101,7 +116,7 @@ export const DateTimePicker = ({ }; const handleDateChange = (newDate: DateTime | null) => { - setDateTime((prev) => + setSelectedDateTime((prev) => newDate ? newDate.set({ hour: prev?.hour || 0, @@ -113,7 +128,7 @@ export const DateTimePicker = ({ const handleTimeChange = (newTime: DateTime | null) => { if (newTime) { - setDateTime((prev) => { + setSelectedDateTime((prev) => { if (prev) { // Ensure hour and minute are valid numbers const newHour = newTime.hour; @@ -127,13 +142,15 @@ export const DateTimePicker = ({ return prev; }); } + if (timeSelectProps.onChange && newTime) { + timeSelectProps.onChange(newTime.toISOTime()); + } }; - const handleTimezoneChange = (newTimezone: string) => { - setTimezone(newTimezone); - setDateTime((prev) => (prev ? prev.setZone(newTimezone) : null)); - if (timeZoneSelectProps?.onChange) { - timeZoneSelectProps.onChange(newTimezone); + const handleTimeZoneChange = (newTimeZone: string) => { + setSelectedTimeZone(newTimeZone); + if (timeZoneSelectProps.onChange) { + timeZoneSelectProps.onChange(newTimeZone); } }; @@ -142,9 +159,11 @@ export const DateTimePicker = ({ ({ + sx={(theme: Theme) => ({ '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { justifyContent: 'space-between', }, @@ -187,7 +206,7 @@ export const DateTimePicker = ({ borderWidth: '0px', })} onChange={handleDateChange} - value={dateTime} + value={selectedDateTime} {...dateCalendarProps} /> )} @@ -217,8 +236,8 @@ export const DateTimePicker = ({ )} @@ -236,7 +255,7 @@ export const DateTimePicker = ({ label: 'Cancel', onClick: handleClose, }} - sx={(theme) => ({ + sx={(theme: Theme) => ({ marginBottom: theme.spacing(1), marginRight: theme.spacing(2), })} From 5f35d7082e4aa0b8fbbf60f8a1625b45073f13de Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 3 Dec 2024 19:25:50 -0600 Subject: [PATCH 21/45] PR feedback --- .../components/DatePicker/DateTimePicker.tsx | 4 +- .../DatePicker/DateTimeRangePicker.tsx | 51 ++++++++++++++----- 2 files changed, 40 insertions(+), 15 deletions(-) diff --git a/packages/ui/src/components/DatePicker/DateTimePicker.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.tsx index 4d65dc6f1b9..ebe97d07071 100644 --- a/packages/ui/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/ui/src/components/DatePicker/DateTimePicker.tsx @@ -161,9 +161,7 @@ export const DateTimePicker = ({ value={ selectedDateTime ? `${selectedDateTime.toFormat(format)}${ - showTimeZone && selectedTimeZone - ? ` (${selectedTimeZone})` - : '' + selectedTimeZone ? ` (${selectedTimeZone})` : '' }` : '' } diff --git a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx index ed57af7c37c..49b155b9e39 100644 --- a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx @@ -12,11 +12,19 @@ interface DateTimeRangePickerProps { /** Format for displaying the date-time */ format?: string; /** Callback when the date-time range changes */ - onChange?: (start: DateTime | null, end: DateTime | null) => void; + onChange?: ( + start: DateTime | null, + end: DateTime | null, + startTimeZone?: null | string + ) => void; + /** Whether to show the timezone field for the end date picker */ + showEndTimeZone?: boolean; /** Initial or controlled value for the start date-time */ startDateTimeValue?: DateTime | null; /** Custom labels for the start and end date/time fields */ startLabel?: string; + /** Initial or controlled value for the start timezone */ + startTimeZoneValue?: null | string; } export const DateTimeRangePicker = ({ @@ -25,7 +33,8 @@ export const DateTimeRangePicker = ({ format = 'yyyy-MM-dd HH:mm', onChange, startDateTimeValue = null, - startLabel = 'Start Date', + startLabel = 'Start Date and Time', + startTimeZoneValue = null, }: DateTimeRangePickerProps) => { const [startDateTime, setStartDateTime] = useState( startDateTimeValue @@ -33,54 +42,72 @@ export const DateTimeRangePicker = ({ const [endDateTime, setEndDateTime] = useState( endDateTimeValue ); + const [startTimeZone, setStartTimeZone] = useState( + startTimeZoneValue + ); const [error, setError] = useState(); const handleStartDateTimeChange = (newStart: DateTime | null) => { setStartDateTime(newStart); - // Reset error if the selection is valid if (endDateTime && newStart && endDateTime >= newStart) { - setError(undefined); + setError(undefined); // Clear error if valid } if (onChange) { - onChange(newStart, endDateTime); + onChange(newStart, endDateTime, startTimeZone); } }; const handleEndDateTimeChange = (newEnd: DateTime | null) => { - // Check if the end date is before the start date if (startDateTime && newEnd && newEnd < startDateTime) { - setError('Invalid date and time'); + setError('End date/time must be after the start date/time.'); } else { setEndDateTime(newEnd); - setError(undefined); // Clear the error if the selection is valid + setError(undefined); + } + + if (onChange) { + onChange(startDateTime, newEnd, startTimeZone); } + }; + + const handleStartTimeZoneChange = (newTimeZone: null | string) => { + setStartTimeZone(newTimeZone); if (onChange) { - onChange(startDateTime, newEnd); + onChange(startDateTime, endDateTime, newTimeZone); } }; return ( + {/* Start DateTime Picker */} + + {/* End DateTime Picker */} From 208b2a07a2b35361a7aa14ed7b5101ccd29d9a03 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 3 Dec 2024 19:27:51 -0600 Subject: [PATCH 22/45] code cleanup --- packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx index 49b155b9e39..f4cb77b175a 100644 --- a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx @@ -32,6 +32,7 @@ export const DateTimeRangePicker = ({ endLabel = 'End Date and Time', format = 'yyyy-MM-dd HH:mm', onChange, + showEndTimeZone = false, startDateTimeValue = null, startLabel = 'Start Date and Time', startTimeZoneValue = null, @@ -106,7 +107,7 @@ export const DateTimeRangePicker = ({ format={format} label={endLabel} onChange={handleEndDateTimeChange} - showTimeZone={false} + showTimeZone={showEndTimeZone} timeSelectProps={{ label: 'End Time' }} value={endDateTime} /> From d1c3fa1528d70ebef0508b4888a297965b9768c8 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 4 Dec 2024 09:24:03 -0600 Subject: [PATCH 23/45] Move DatePicker back to src/components --- .../DatePicker/DatePicker.stories.tsx | 43 +++ .../components/DatePicker/DatePicker.test.tsx | 80 ++++++ .../src/components/DatePicker/DatePicker.tsx | 92 ++++++ .../DatePicker/DateTimePicker.stories.tsx | 93 ++++++ .../DatePicker/DateTimePicker.test.tsx | 143 ++++++++++ .../components/DatePicker/DateTimePicker.tsx | 265 ++++++++++++++++++ .../DateTimeRangePicker.stories.tsx | 77 +++++ .../DatePicker/DateTimeRangePicker.test.tsx | 59 ++++ .../DatePicker/DateTimeRangePicker.tsx | 116 ++++++++ .../components/DatePicker/TimeZoneSelect.tsx | 64 +++++ packages/ui/src/components/index.ts | 1 - 11 files changed, 1032 insertions(+), 1 deletion(-) create mode 100644 packages/manager/src/components/DatePicker/DatePicker.stories.tsx create mode 100644 packages/manager/src/components/DatePicker/DatePicker.test.tsx create mode 100644 packages/manager/src/components/DatePicker/DatePicker.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.test.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimePicker.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx create mode 100644 packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx create mode 100644 packages/manager/src/components/DatePicker/TimeZoneSelect.tsx diff --git a/packages/manager/src/components/DatePicker/DatePicker.stories.tsx b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx new file mode 100644 index 00000000000..db3925b03eb --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx @@ -0,0 +1,43 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DatePicker } from './DatePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const ControlledExample: Story = { + render: () => { + const ControlledDatePicker = () => { + const [selectedDate, setSelectedDate] = React.useState( + DateTime.now() + ); + + const handleChange = (newDate: DateTime | null) => { + setSelectedDate(newDate); + action('Controlled date change')(newDate?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +const meta: Meta = { + component: DatePicker, + title: 'Components/DatePicker', +}; + +export default meta; diff --git a/packages/manager/src/components/DatePicker/DatePicker.test.tsx b/packages/manager/src/components/DatePicker/DatePicker.test.tsx new file mode 100644 index 00000000000..e051d160ec5 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.test.tsx @@ -0,0 +1,80 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DatePicker } from './DatePicker'; + +import type { DatePickerProps } from './DatePicker'; + +const props: DatePickerProps = { + onChange: vi.fn(), + placeholder: 'Pick a date', + textFieldProps: { errorText: 'Invalid date', label: 'Select a date' }, + value: null, +}; + +describe('DatePicker', () => { + it('should render the DatePicker component', () => { + renderWithTheme(); + const DatePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + + expect(DatePickerField).toBeVisible(); + }); + + it('should handle value changes', async () => { + renderWithTheme(); + + const calendarButton = screen.getByRole('button', { name: 'Choose date' }); + + // Click the calendar button to open the date picker + await userEvent.click(calendarButton); + + // Find a date button to click (e.g., the 15th of the month) + const dateToSelect = screen.getByRole('gridcell', { name: '15' }); + await userEvent.click(dateToSelect); + + // Check if onChange was called after selecting a date + expect(props.onChange).toHaveBeenCalled(); + }); + + it('should display the error text when provided', () => { + renderWithTheme(); + const errorMessage = screen.getByText('Invalid date'); + expect(errorMessage).toBeVisible(); + }); + + it('should display the helper text when provided', () => { + renderWithTheme(); + const helperText = screen.getByText('Choose a valid date'); + expect(helperText).toBeVisible(); + }); + + it('should use the default format when no format is specified', () => { + renderWithTheme( + + ); + const datePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + expect(datePickerField).toHaveValue('2024-10-25'); + }); + + it('should handle the custom format correctly', () => { + renderWithTheme( + + ); + const datePickerField = screen.getByRole('textbox', { + name: 'Select a date', + }); + expect(datePickerField).toHaveValue('25/10/2024'); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx new file mode 100644 index 00000000000..a3a5a722e82 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -0,0 +1,92 @@ +import { TextField } from '@linode/ui'; +import { useTheme } from '@mui/material/styles'; +import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; +import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; +import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; +import React from 'react'; + +import type { TextFieldProps } from '@linode/ui'; +import type { DatePickerProps as MuiDatePickerProps } from '@mui/x-date-pickers/DatePicker'; +import type { DateTime } from 'luxon'; + +export interface DatePickerProps + extends Omit, 'onChange' | 'value'> { + /** Error text to display below the input */ + errorText?: string; + /** Format of the date when rendered in the input field. */ + format?: string; + /** Helper text to display below the input */ + helperText?: string; + /** Label to display for the date picker input */ + label?: string; + /** Callback function fired when the value changes */ + onChange?: (newDate: DateTime | null) => void; + /** Placeholder text for the date picker input */ + placeholder?: string; + /** Additional props to pass to the underlying TextField component */ + textFieldProps?: Omit; + /** The currently selected date */ + value?: DateTime | null; +} + +export const DatePicker = ({ + format = 'yyyy-MM-dd', + helperText = '', + label = 'Select a date', + onChange, + placeholder = 'Pick a date', + textFieldProps, + value = null, + ...props +}: DatePickerProps) => { + const theme = useTheme(); + + const onChangeHandler = (newDate: DateTime | null) => { + if (onChange) { + onChange(newDate); + } + }; + + return ( + + + + ); +}; diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx new file mode 100644 index 00000000000..32d65dfa0b7 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx @@ -0,0 +1,93 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DateTimePicker } from './DateTimePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const ControlledExample: Story = { + render: () => { + const ControlledDateTimePicker = () => { + const [ + selectedDateTime, + setSelectedDateTime, + ] = React.useState(DateTime.now()); + + const handleChange = (newDateTime: DateTime | null) => { + setSelectedDateTime(newDateTime); + action('Controlled dateTime change')(newDateTime?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +export const DefaultExample: Story = { + args: { + label: 'Default Date-Time Picker', + onApply: action('Apply clicked'), + onCancel: action('Cancel clicked'), + onChange: action('Date-Time selected'), + placeholder: 'yyyy-MM-dd HH:mm', + }, +}; + +export const WithErrorText: Story = { + args: { + errorText: 'This field is required', + label: 'Date-Time Picker with Error', + onApply: action('Apply clicked with error'), + onCancel: action('Cancel clicked with error'), + onChange: action('Date-Time selected with error'), + placeholder: 'yyyy-MM-dd HH:mm', + }, +}; + +const meta: Meta = { + argTypes: { + errorText: { + control: { type: 'text' }, + }, + format: { + control: { type: 'text' }, + }, + label: { + control: { type: 'text' }, + }, + onApply: { action: 'applyClicked' }, + onCancel: { action: 'cancelClicked' }, + onChange: { action: 'dateTimeChanged' }, + placeholder: { + control: { type: 'text' }, + }, + }, + args: { + format: 'yyyy-MM-dd HH:mm', + label: 'Date-Time Picker', + placeholder: 'Select a date and time', + }, + component: DateTimePicker, + title: 'Components/DateTimePicker', +}; + +export default meta; diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx new file mode 100644 index 00000000000..12f1795a747 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.test.tsx @@ -0,0 +1,143 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DateTimePicker } from './DateTimePicker'; + +import type { DateTimePickerProps } from './DateTimePicker'; + +const defaultProps: DateTimePickerProps = { + label: 'Select Date and Time', + onApply: vi.fn(), + onCancel: vi.fn(), + onChange: vi.fn(), + placeholder: 'yyyy-MM-dd HH:mm', + value: DateTime.fromISO('2024-10-25T15:30:00'), +}; + +describe('DateTimePicker Component', () => { + it('should render the DateTimePicker component with the correct label and placeholder', () => { + renderWithTheme(); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + expect(textField).toBeVisible(); + expect(textField).toHaveAttribute('placeholder', 'yyyy-MM-dd HH:mm'); + }); + + it('should open the Popover when the TextField is clicked', async () => { + renderWithTheme(); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + await userEvent.click(textField); + expect(screen.getByRole('dialog')).toBeVisible(); // Verifying the Popover is open + }); + + it('should call onCancel when the Cancel button is clicked', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const cancelButton = screen.getByRole('button', { name: /Cancel/i }); + await userEvent.click(cancelButton); + expect(defaultProps.onCancel).toHaveBeenCalled(); + }); + + it('should call onApply when the Apply button is clicked', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const applyButton = screen.getByRole('button', { name: /Apply/i }); + await userEvent.click(applyButton); + expect(defaultProps.onApply).toHaveBeenCalled(); + expect(defaultProps.onChange).toHaveBeenCalledWith(expect.any(DateTime)); // Ensuring onChange was called with a DateTime object + }); + + it('should handle date changes correctly', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + + // Simulate selecting a date (e.g., 15th of the month) + const dateButton = screen.getByRole('gridcell', { name: '15' }); + await userEvent.click(dateButton); + + // Check that the displayed value has been updated correctly (this assumes the date format) + expect(defaultProps.onChange).toHaveBeenCalled(); + }); + + it('should handle timezone changes correctly', async () => { + const timezoneChangeMock = vi.fn(); // Create a mock function + + const updatedProps = { + ...defaultProps, + timeZoneSelectProps: { onChange: timezoneChangeMock, value: 'UTC' }, + }; + + renderWithTheme(); + + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + + // Simulate selecting a timezone from the TimeZoneSelect + const timezoneInput = screen.getByPlaceholderText(/Choose a Timezone/i); + await userEvent.click(timezoneInput); + + // Select a timezone from the dropdown options + await userEvent.click( + screen.getByRole('option', { name: '(GMT -11:00) Niue Time' }) + ); + + // Click the Apply button to trigger the change + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Verify that the onChange function was called with the expected value + expect(timezoneChangeMock).toHaveBeenCalledWith('Pacific/Niue'); + }); + + it('should display the error text when provided', () => { + renderWithTheme( + + ); + expect(screen.getByText(/Invalid date-time/i)).toBeVisible(); + }); + + it('should format the date-time correctly when a custom format is provided', () => { + renderWithTheme( + + ); + const textField = screen.getByRole('textbox', { + name: 'Select Date and Time', + }); + + expect(textField).toHaveValue('25/10/2024 15:30'); + }); + it('should not render the time selector when showTime is false', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const timePicker = screen.queryByLabelText(/Select Time/i); // Label from timeSelectProps + expect(timePicker).not.toBeInTheDocument(); + }); + + it('should not render the timezone selector when showTimeZone is false', async () => { + renderWithTheme(); + await userEvent.click( + screen.getByRole('textbox', { name: 'Select Date and Time' }) + ); + const timeZoneSelect = screen.queryByLabelText(/Timezone/i); // Label from timeZoneSelectProps + expect(timeZoneSelect).not.toBeInTheDocument(); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx new file mode 100644 index 00000000000..027b25822d2 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -0,0 +1,265 @@ +import { TextField } from '@linode/ui'; +import { Divider } from '@linode/ui'; +import { Box } from '@linode/ui'; +import { Grid, Popover } from '@mui/material'; +import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; +import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; +import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; +import { TimePicker } from '@mui/x-date-pickers/TimePicker'; +import React, { useEffect, useState } from 'react'; + +import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; + +import { TimeZoneSelect } from './TimeZoneSelect'; + +import type { TextFieldProps } from '@linode/ui'; +import type { Theme } from '@mui/material/styles'; +import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; +import type { DateTime } from 'luxon'; + +export interface DateTimePickerProps { + /** Additional props for the DateCalendar */ + dateCalendarProps?: Partial>; + /** Error text for the date picker field */ + errorText?: string; + /** Format for displaying the date-time */ + format?: string; + /** Label for the input field */ + label?: string; + /** Callback when the "Apply" button is clicked */ + onApply?: () => void; + /** Callback when the "Cancel" button is clicked */ + onCancel?: () => void; + /** Callback when date-time changes */ + onChange?: (dateTime: DateTime | null) => void; + /** Placeholder text for the input field */ + placeholder?: string; + /** Whether to show the time selector */ + showTime?: boolean; + /** Whether to show the timezone selector */ + showTimeZone?: boolean; + /** Props for customizing the TimePicker component */ + timeSelectProps?: { + label?: string; + onChange?: (time: null | string) => void; + value?: null | string; + }; + /** Props for customizing the TimeZoneSelect component */ + timeZoneSelectProps?: { + label?: string; + onChange?: (timezone: string) => void; + value?: null | string; + }; + /** Initial or controlled dateTime value */ + value?: DateTime | null; +} + +export const DateTimePicker = ({ + dateCalendarProps = {}, + errorText = '', + format = 'yyyy-MM-dd HH:mm', + label = 'Select Date and Time', + onApply, + onCancel, + onChange, + placeholder = 'yyyy-MM-dd HH:mm', + showTime = true, + showTimeZone = true, + timeSelectProps = {}, + timeZoneSelectProps = {}, + value = null, +}: DateTimePickerProps) => { + const [anchorEl, setAnchorEl] = useState(null); + const [selectedDateTime, setSelectedDateTime] = useState( + value + ); + const [selectedTimeZone, setSelectedTimeZone] = useState( + timeZoneSelectProps.value || null + ); + + const TimePickerFieldProps: TextFieldProps = { + label: timeSelectProps?.label ?? 'Select Time', + noMarginTop: true, + }; + + useEffect(() => { + if (value) { + setSelectedDateTime(value); + } + }, [value]); + + useEffect(() => { + if (timeZoneSelectProps.value) { + setSelectedTimeZone(timeZoneSelectProps.value); + } + }, [timeZoneSelectProps.value]); + + const handleOpen = (event: React.MouseEvent) => { + setAnchorEl(event.currentTarget); + }; + + const handleClose = () => { + setAnchorEl(null); + if (onCancel) { + onCancel(); + } + }; + + const handleApply = () => { + setAnchorEl(null); + if (onChange) { + onChange(selectedDateTime); + } + if (onApply) { + onApply(); + } + }; + + const handleDateChange = (newDate: DateTime | null) => { + setSelectedDateTime((prev) => + newDate + ? newDate.set({ + hour: prev?.hour || 0, + minute: prev?.minute || 0, + }) + : null + ); + }; + + const handleTimeChange = (newTime: DateTime | null) => { + if (newTime) { + setSelectedDateTime((prev) => { + if (prev) { + // Ensure hour and minute are valid numbers + const newHour = newTime.hour; + const newMinute = newTime.minute; + + if (typeof newHour === 'number' && typeof newMinute === 'number') { + return prev.set({ hour: newHour, minute: newMinute }); + } + } + // Return the current `prev` value if newTime is invalid + return prev; + }); + } + if (timeSelectProps.onChange && newTime) { + timeSelectProps.onChange(newTime.toISOTime()); + } + }; + + const handleTimeZoneChange = (newTimeZone: string) => { + setSelectedTimeZone(newTimeZone); + if (timeZoneSelectProps.onChange) { + timeZoneSelectProps.onChange(newTimeZone); + } + }; + + return ( + + + + + + + ({ + '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { + justifyContent: 'space-between', + }, + '& .MuiDayCalendar-weekDayLabel': { + fontSize: '0.875rem', + }, + '& .MuiPickersCalendarHeader-label': { + fontFamily: theme.font.bold, + }, + '& .MuiPickersCalendarHeader-root': { + borderBottom: `1px solid ${theme.borderColors.divider}`, + fontSize: '0.875rem', + paddingBottom: theme.spacing(1), + }, + '& .MuiPickersDay-root': { + fontSize: '0.875rem', + margin: `${theme.spacing(0.5)}px`, + }, + borderRadius: `${theme.spacing(2)}`, + borderWidth: '0px', + })} + onChange={handleDateChange} + value={selectedDateTime} + {...dateCalendarProps} + /> + + {showTime && ( + + + + )} + {showTimeZone && ( + + + + )} + + + + + ({ + marginBottom: theme.spacing(1), + marginRight: theme.spacing(2), + })} + /> + + + + ); +}; diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx new file mode 100644 index 00000000000..70ad6b2a39d --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx @@ -0,0 +1,77 @@ +import { action } from '@storybook/addon-actions'; +import { DateTime } from 'luxon'; +import * as React from 'react'; + +import { DateTimeRangePicker } from './DateTimeRangePicker'; + +import type { Meta, StoryObj } from '@storybook/react'; + +type Story = StoryObj; + +export const Default: Story = { + args: { + endLabel: 'End Date and Time', + format: 'yyyy-MM-dd HH:mm', + onChange: action('DateTime range changed'), + startLabel: 'Start Date and Time', + }, + render: (args) => , +}; + +export const WithInitialValues: Story = { + render: () => { + const ComponentWithState: React.FC = () => { + const [start, setStart] = React.useState( + DateTime.now().minus({ days: 1 }) + ); + const [end, setEnd] = React.useState(DateTime.now()); + + const handleDateChange = ( + newStart: DateTime | null, + newEnd: DateTime | null + ) => { + setStart(newStart); + setEnd(newEnd); + action('DateTime range changed')(newStart?.toISO(), newEnd?.toISO()); + }; + + return ( + + ); + }; + + return ; + }, +}; + +const meta: Meta = { + argTypes: { + endLabel: { + control: 'text', + description: 'Label for the end date picker', + }, + format: { + control: 'text', + description: 'Format for displaying the date-time', + }, + onChange: { + action: 'DateTime range changed', + description: 'Callback when the date-time range changes', + }, + startLabel: { + control: 'text', + description: 'Label for the start date picker', + }, + }, + component: DateTimeRangePicker, + title: 'Components/DateTimeRangePicker', +}; + +export default meta; diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx new file mode 100644 index 00000000000..f3421b684c3 --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -0,0 +1,59 @@ +import { screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; +import * as React from 'react'; + +import { renderWithTheme } from 'src/utilities/testHelpers'; + +import { DateTimeRangePicker } from './DateTimeRangePicker'; + +describe('DateTimeRangePicker Component', () => { + const onChangeMock = vi.fn(); + + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should render start and end DateTimePickers with correct labels', () => { + renderWithTheme(); + + expect(screen.getByLabelText('Start Date')).toBeVisible(); + expect(screen.getByLabelText('End Date and Time')).toBeVisible(); + }); + + it('should call onChange when start date is changed', async () => { + renderWithTheme(); + + // Open start date picker + await userEvent.click(screen.getByLabelText('Start Date')); + + await userEvent.click(screen.getByRole('gridcell', { name: '10' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Check if the onChange function is called with the expected DateTime value + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ day: 10 }), + null + ); + }); + + it('should show error when end date-time is before start date-time', async () => { + renderWithTheme(); + + // Set start date-time to the 15th + const startDateField = screen.getByLabelText('Start Date'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Open the end date picker + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + + // Check if the date before the start date is disabled via a class or attribute + const invalidDate = screen.getByRole('gridcell', { name: '10' }); + expect(invalidDate).toHaveClass('Mui-disabled'); // or check for the specific attribute used + + // Confirm error message is not shown since the click was blocked + expect(screen.queryByText('Invalid date and Time')).not.toBeInTheDocument(); + }); +}); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx new file mode 100644 index 00000000000..f4cb77b175a --- /dev/null +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -0,0 +1,116 @@ +import { Box } from '@linode/ui'; +import { DateTime } from 'luxon'; +import React, { useState } from 'react'; + +import { DateTimePicker } from './DateTimePicker'; + +interface DateTimeRangePickerProps { + /** Initial or controlled value for the end date-time */ + endDateTimeValue?: DateTime | null; + /** Custom labels for the start and end date/time fields */ + endLabel?: string; + /** Format for displaying the date-time */ + format?: string; + /** Callback when the date-time range changes */ + onChange?: ( + start: DateTime | null, + end: DateTime | null, + startTimeZone?: null | string + ) => void; + /** Whether to show the timezone field for the end date picker */ + showEndTimeZone?: boolean; + /** Initial or controlled value for the start date-time */ + startDateTimeValue?: DateTime | null; + /** Custom labels for the start and end date/time fields */ + startLabel?: string; + /** Initial or controlled value for the start timezone */ + startTimeZoneValue?: null | string; +} + +export const DateTimeRangePicker = ({ + endDateTimeValue = null, + endLabel = 'End Date and Time', + format = 'yyyy-MM-dd HH:mm', + onChange, + showEndTimeZone = false, + startDateTimeValue = null, + startLabel = 'Start Date and Time', + startTimeZoneValue = null, +}: DateTimeRangePickerProps) => { + const [startDateTime, setStartDateTime] = useState( + startDateTimeValue + ); + const [endDateTime, setEndDateTime] = useState( + endDateTimeValue + ); + const [startTimeZone, setStartTimeZone] = useState( + startTimeZoneValue + ); + const [error, setError] = useState(); + + const handleStartDateTimeChange = (newStart: DateTime | null) => { + setStartDateTime(newStart); + + if (endDateTime && newStart && endDateTime >= newStart) { + setError(undefined); // Clear error if valid + } + + if (onChange) { + onChange(newStart, endDateTime, startTimeZone); + } + }; + + const handleEndDateTimeChange = (newEnd: DateTime | null) => { + if (startDateTime && newEnd && newEnd < startDateTime) { + setError('End date/time must be after the start date/time.'); + } else { + setEndDateTime(newEnd); + setError(undefined); + } + + if (onChange) { + onChange(startDateTime, newEnd, startTimeZone); + } + }; + + const handleStartTimeZoneChange = (newTimeZone: null | string) => { + setStartTimeZone(newTimeZone); + + if (onChange) { + onChange(startDateTime, endDateTime, newTimeZone); + } + }; + + return ( + + {/* Start DateTime Picker */} + + + {/* End DateTime Picker */} + + + ); +}; diff --git a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx new file mode 100644 index 00000000000..19312098007 --- /dev/null +++ b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx @@ -0,0 +1,64 @@ +import { DateTime } from 'luxon'; +import React from 'react'; + +import { timezones } from 'src/assets/timezones/timezones'; +import { Autocomplete } from 'src/components/Autocomplete/Autocomplete'; + +type Timezone = typeof timezones[number]; + +interface TimeZoneSelectProps { + disabled?: boolean; + errorText?: string; + label?: string; + noMarginTop?: boolean; + onChange: (timezone: string) => void; + value: null | string; +} + +const getOptionLabel = ({ label, offset }: Timezone) => { + const minutes = (Math.abs(offset) % 60).toLocaleString(undefined, { + minimumIntegerDigits: 2, + useGrouping: false, + }); + const hours = Math.floor(Math.abs(offset) / 60); + const isPositive = Math.abs(offset) === offset ? '+' : '-'; + + return `(GMT ${isPositive}${hours}:${minutes}) ${label}`; +}; + +const getTimezoneOptions = () => { + return timezones + .map((tz) => { + const offset = DateTime.now().setZone(tz.name).offset; + const label = getOptionLabel({ ...tz, offset }); + return { label, offset, value: tz.name }; + }) + .sort((a, b) => a.offset - b.offset); +}; + +const timezoneOptions = getTimezoneOptions(); + +export const TimeZoneSelect = ({ + disabled = false, + errorText, + label = 'Timezone', + noMarginTop = false, + onChange, + value, +}: TimeZoneSelectProps) => { + return ( + option.value === value) ?? undefined + } + autoHighlight + disabled={disabled} + errorText={errorText} + label={label} + noMarginTop={noMarginTop} + onChange={(e, option) => onChange(option?.value || '')} + options={timezoneOptions} + placeholder="Choose a Timezone" + /> + ); +}; diff --git a/packages/ui/src/components/index.ts b/packages/ui/src/components/index.ts index f64e7b4105a..db690f6940e 100644 --- a/packages/ui/src/components/index.ts +++ b/packages/ui/src/components/index.ts @@ -26,4 +26,3 @@ export * from './Tooltip'; export * from './TooltipIcon'; export * from './Typography'; export * from './VisibilityTooltip'; -export * from './DatePicker'; From 6186d267968389e09591eb4e497248b07391288c Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 4 Dec 2024 10:11:33 -0600 Subject: [PATCH 24/45] Reverting changes --- packages/ui/package.json | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/packages/ui/package.json b/packages/ui/package.json index 3a85d11a12a..a0926371058 100644 --- a/packages/ui/package.json +++ b/packages/ui/package.json @@ -20,9 +20,7 @@ "@emotion/styled": "^11.11.0", "@mui/material": "^5.14.7", "react": "^18.2.0", - "react-dom": "^18.2.0", - "@mui/x-date-pickers": "^7.12.0", - "luxon": "3.4.4" + "react-dom": "^18.2.0" }, "scripts": { "start": "tsc -w --preserveWatchOutput", From 2cd83660d85c758088aa3b1e2c8c86f9d19a7045 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 4 Dec 2024 10:30:39 -0600 Subject: [PATCH 25/45] Code cleanup --- .../DatePicker/DatePicker.stories.tsx | 43 --- .../components/DatePicker/DatePicker.test.tsx | 80 ------ .../src/components/DatePicker/DatePicker.tsx | 92 ------ .../DatePicker/DateTimePicker.stories.tsx | 93 ------ .../DatePicker/DateTimePicker.test.tsx | 143 ---------- .../components/DatePicker/DateTimePicker.tsx | 265 ------------------ .../DateTimeRangePicker.stories.tsx | 77 ----- .../DatePicker/DateTimeRangePicker.test.tsx | 59 ---- .../DatePicker/DateTimeRangePicker.tsx | 116 -------- .../components/DatePicker/TimeZoneSelect.tsx | 64 ----- .../ui/src/components/DatePicker/index.ts | 1 - 11 files changed, 1033 deletions(-) delete mode 100644 packages/ui/src/components/DatePicker/DatePicker.stories.tsx delete mode 100644 packages/ui/src/components/DatePicker/DatePicker.test.tsx delete mode 100644 packages/ui/src/components/DatePicker/DatePicker.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimePicker.test.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimePicker.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx delete mode 100644 packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx delete mode 100644 packages/ui/src/components/DatePicker/TimeZoneSelect.tsx delete mode 100644 packages/ui/src/components/DatePicker/index.ts diff --git a/packages/ui/src/components/DatePicker/DatePicker.stories.tsx b/packages/ui/src/components/DatePicker/DatePicker.stories.tsx deleted file mode 100644 index db3925b03eb..00000000000 --- a/packages/ui/src/components/DatePicker/DatePicker.stories.tsx +++ /dev/null @@ -1,43 +0,0 @@ -import { action } from '@storybook/addon-actions'; -import { DateTime } from 'luxon'; -import * as React from 'react'; - -import { DatePicker } from './DatePicker'; - -import type { Meta, StoryObj } from '@storybook/react'; - -type Story = StoryObj; - -export const ControlledExample: Story = { - render: () => { - const ControlledDatePicker = () => { - const [selectedDate, setSelectedDate] = React.useState( - DateTime.now() - ); - - const handleChange = (newDate: DateTime | null) => { - setSelectedDate(newDate); - action('Controlled date change')(newDate?.toISO()); - }; - - return ( - - ); - }; - - return ; - }, -}; - -const meta: Meta = { - component: DatePicker, - title: 'Components/DatePicker', -}; - -export default meta; diff --git a/packages/ui/src/components/DatePicker/DatePicker.test.tsx b/packages/ui/src/components/DatePicker/DatePicker.test.tsx deleted file mode 100644 index e051d160ec5..00000000000 --- a/packages/ui/src/components/DatePicker/DatePicker.test.tsx +++ /dev/null @@ -1,80 +0,0 @@ -import { screen } from '@testing-library/react'; -import userEvent from '@testing-library/user-event'; -import { DateTime } from 'luxon'; -import * as React from 'react'; - -import { renderWithTheme } from 'src/utilities/testHelpers'; - -import { DatePicker } from './DatePicker'; - -import type { DatePickerProps } from './DatePicker'; - -const props: DatePickerProps = { - onChange: vi.fn(), - placeholder: 'Pick a date', - textFieldProps: { errorText: 'Invalid date', label: 'Select a date' }, - value: null, -}; - -describe('DatePicker', () => { - it('should render the DatePicker component', () => { - renderWithTheme(); - const DatePickerField = screen.getByRole('textbox', { - name: 'Select a date', - }); - - expect(DatePickerField).toBeVisible(); - }); - - it('should handle value changes', async () => { - renderWithTheme(); - - const calendarButton = screen.getByRole('button', { name: 'Choose date' }); - - // Click the calendar button to open the date picker - await userEvent.click(calendarButton); - - // Find a date button to click (e.g., the 15th of the month) - const dateToSelect = screen.getByRole('gridcell', { name: '15' }); - await userEvent.click(dateToSelect); - - // Check if onChange was called after selecting a date - expect(props.onChange).toHaveBeenCalled(); - }); - - it('should display the error text when provided', () => { - renderWithTheme(); - const errorMessage = screen.getByText('Invalid date'); - expect(errorMessage).toBeVisible(); - }); - - it('should display the helper text when provided', () => { - renderWithTheme(); - const helperText = screen.getByText('Choose a valid date'); - expect(helperText).toBeVisible(); - }); - - it('should use the default format when no format is specified', () => { - renderWithTheme( - - ); - const datePickerField = screen.getByRole('textbox', { - name: 'Select a date', - }); - expect(datePickerField).toHaveValue('2024-10-25'); - }); - - it('should handle the custom format correctly', () => { - renderWithTheme( - - ); - const datePickerField = screen.getByRole('textbox', { - name: 'Select a date', - }); - expect(datePickerField).toHaveValue('25/10/2024'); - }); -}); diff --git a/packages/ui/src/components/DatePicker/DatePicker.tsx b/packages/ui/src/components/DatePicker/DatePicker.tsx deleted file mode 100644 index a3a5a722e82..00000000000 --- a/packages/ui/src/components/DatePicker/DatePicker.tsx +++ /dev/null @@ -1,92 +0,0 @@ -import { TextField } from '@linode/ui'; -import { useTheme } from '@mui/material/styles'; -import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; -import { DatePicker as MuiDatePicker } from '@mui/x-date-pickers/DatePicker'; -import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; -import React from 'react'; - -import type { TextFieldProps } from '@linode/ui'; -import type { DatePickerProps as MuiDatePickerProps } from '@mui/x-date-pickers/DatePicker'; -import type { DateTime } from 'luxon'; - -export interface DatePickerProps - extends Omit, 'onChange' | 'value'> { - /** Error text to display below the input */ - errorText?: string; - /** Format of the date when rendered in the input field. */ - format?: string; - /** Helper text to display below the input */ - helperText?: string; - /** Label to display for the date picker input */ - label?: string; - /** Callback function fired when the value changes */ - onChange?: (newDate: DateTime | null) => void; - /** Placeholder text for the date picker input */ - placeholder?: string; - /** Additional props to pass to the underlying TextField component */ - textFieldProps?: Omit; - /** The currently selected date */ - value?: DateTime | null; -} - -export const DatePicker = ({ - format = 'yyyy-MM-dd', - helperText = '', - label = 'Select a date', - onChange, - placeholder = 'Pick a date', - textFieldProps, - value = null, - ...props -}: DatePickerProps) => { - const theme = useTheme(); - - const onChangeHandler = (newDate: DateTime | null) => { - if (onChange) { - onChange(newDate); - } - }; - - return ( - - - - ); -}; diff --git a/packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx deleted file mode 100644 index 32d65dfa0b7..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimePicker.stories.tsx +++ /dev/null @@ -1,93 +0,0 @@ -import { action } from '@storybook/addon-actions'; -import { DateTime } from 'luxon'; -import * as React from 'react'; - -import { DateTimePicker } from './DateTimePicker'; - -import type { Meta, StoryObj } from '@storybook/react'; - -type Story = StoryObj; - -export const ControlledExample: Story = { - render: () => { - const ControlledDateTimePicker = () => { - const [ - selectedDateTime, - setSelectedDateTime, - ] = React.useState(DateTime.now()); - - const handleChange = (newDateTime: DateTime | null) => { - setSelectedDateTime(newDateTime); - action('Controlled dateTime change')(newDateTime?.toISO()); - }; - - return ( - - ); - }; - - return ; - }, -}; - -export const DefaultExample: Story = { - args: { - label: 'Default Date-Time Picker', - onApply: action('Apply clicked'), - onCancel: action('Cancel clicked'), - onChange: action('Date-Time selected'), - placeholder: 'yyyy-MM-dd HH:mm', - }, -}; - -export const WithErrorText: Story = { - args: { - errorText: 'This field is required', - label: 'Date-Time Picker with Error', - onApply: action('Apply clicked with error'), - onCancel: action('Cancel clicked with error'), - onChange: action('Date-Time selected with error'), - placeholder: 'yyyy-MM-dd HH:mm', - }, -}; - -const meta: Meta = { - argTypes: { - errorText: { - control: { type: 'text' }, - }, - format: { - control: { type: 'text' }, - }, - label: { - control: { type: 'text' }, - }, - onApply: { action: 'applyClicked' }, - onCancel: { action: 'cancelClicked' }, - onChange: { action: 'dateTimeChanged' }, - placeholder: { - control: { type: 'text' }, - }, - }, - args: { - format: 'yyyy-MM-dd HH:mm', - label: 'Date-Time Picker', - placeholder: 'Select a date and time', - }, - component: DateTimePicker, - title: 'Components/DateTimePicker', -}; - -export default meta; diff --git a/packages/ui/src/components/DatePicker/DateTimePicker.test.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.test.tsx deleted file mode 100644 index 12f1795a747..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimePicker.test.tsx +++ /dev/null @@ -1,143 +0,0 @@ -import { screen } from '@testing-library/react'; -import userEvent from '@testing-library/user-event'; -import { DateTime } from 'luxon'; -import * as React from 'react'; - -import { renderWithTheme } from 'src/utilities/testHelpers'; - -import { DateTimePicker } from './DateTimePicker'; - -import type { DateTimePickerProps } from './DateTimePicker'; - -const defaultProps: DateTimePickerProps = { - label: 'Select Date and Time', - onApply: vi.fn(), - onCancel: vi.fn(), - onChange: vi.fn(), - placeholder: 'yyyy-MM-dd HH:mm', - value: DateTime.fromISO('2024-10-25T15:30:00'), -}; - -describe('DateTimePicker Component', () => { - it('should render the DateTimePicker component with the correct label and placeholder', () => { - renderWithTheme(); - const textField = screen.getByRole('textbox', { - name: 'Select Date and Time', - }); - expect(textField).toBeVisible(); - expect(textField).toHaveAttribute('placeholder', 'yyyy-MM-dd HH:mm'); - }); - - it('should open the Popover when the TextField is clicked', async () => { - renderWithTheme(); - const textField = screen.getByRole('textbox', { - name: 'Select Date and Time', - }); - await userEvent.click(textField); - expect(screen.getByRole('dialog')).toBeVisible(); // Verifying the Popover is open - }); - - it('should call onCancel when the Cancel button is clicked', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - const cancelButton = screen.getByRole('button', { name: /Cancel/i }); - await userEvent.click(cancelButton); - expect(defaultProps.onCancel).toHaveBeenCalled(); - }); - - it('should call onApply when the Apply button is clicked', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - const applyButton = screen.getByRole('button', { name: /Apply/i }); - await userEvent.click(applyButton); - expect(defaultProps.onApply).toHaveBeenCalled(); - expect(defaultProps.onChange).toHaveBeenCalledWith(expect.any(DateTime)); // Ensuring onChange was called with a DateTime object - }); - - it('should handle date changes correctly', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - - // Simulate selecting a date (e.g., 15th of the month) - const dateButton = screen.getByRole('gridcell', { name: '15' }); - await userEvent.click(dateButton); - - // Check that the displayed value has been updated correctly (this assumes the date format) - expect(defaultProps.onChange).toHaveBeenCalled(); - }); - - it('should handle timezone changes correctly', async () => { - const timezoneChangeMock = vi.fn(); // Create a mock function - - const updatedProps = { - ...defaultProps, - timeZoneSelectProps: { onChange: timezoneChangeMock, value: 'UTC' }, - }; - - renderWithTheme(); - - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - - // Simulate selecting a timezone from the TimeZoneSelect - const timezoneInput = screen.getByPlaceholderText(/Choose a Timezone/i); - await userEvent.click(timezoneInput); - - // Select a timezone from the dropdown options - await userEvent.click( - screen.getByRole('option', { name: '(GMT -11:00) Niue Time' }) - ); - - // Click the Apply button to trigger the change - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - // Verify that the onChange function was called with the expected value - expect(timezoneChangeMock).toHaveBeenCalledWith('Pacific/Niue'); - }); - - it('should display the error text when provided', () => { - renderWithTheme( - - ); - expect(screen.getByText(/Invalid date-time/i)).toBeVisible(); - }); - - it('should format the date-time correctly when a custom format is provided', () => { - renderWithTheme( - - ); - const textField = screen.getByRole('textbox', { - name: 'Select Date and Time', - }); - - expect(textField).toHaveValue('25/10/2024 15:30'); - }); - it('should not render the time selector when showTime is false', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - const timePicker = screen.queryByLabelText(/Select Time/i); // Label from timeSelectProps - expect(timePicker).not.toBeInTheDocument(); - }); - - it('should not render the timezone selector when showTimeZone is false', async () => { - renderWithTheme(); - await userEvent.click( - screen.getByRole('textbox', { name: 'Select Date and Time' }) - ); - const timeZoneSelect = screen.queryByLabelText(/Timezone/i); // Label from timeZoneSelectProps - expect(timeZoneSelect).not.toBeInTheDocument(); - }); -}); diff --git a/packages/ui/src/components/DatePicker/DateTimePicker.tsx b/packages/ui/src/components/DatePicker/DateTimePicker.tsx deleted file mode 100644 index ebe97d07071..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimePicker.tsx +++ /dev/null @@ -1,265 +0,0 @@ -import { TextField } from '@linode/ui'; -import { Divider } from '@linode/ui'; -import { Box } from '@linode/ui'; -import { Grid, Popover } from '@mui/material'; -import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; -import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; -import { LocalizationProvider } from '@mui/x-date-pickers/LocalizationProvider'; -import { TimePicker } from '@mui/x-date-pickers/TimePicker'; -import React, { useEffect, useState } from 'react'; - -import { ActionsPanel } from 'src/components/ActionsPanel/ActionsPanel'; - -import { TimeZoneSelect } from './TimeZoneSelect'; - -import type { TextFieldProps } from '@linode/ui'; -import type { Theme } from '@mui/material/styles'; -import type { DateCalendarProps } from '@mui/x-date-pickers/DateCalendar'; -import type { DateTime } from 'luxon'; - -export interface DateTimePickerProps { - /** Additional props for the DateCalendar */ - dateCalendarProps?: Partial>; - /** Error text for the date picker field */ - errorText?: string; - /** Format for displaying the date-time */ - format?: string; - /** Label for the input field */ - label?: string; - /** Callback when the "Apply" button is clicked */ - onApply?: () => void; - /** Callback when the "Cancel" button is clicked */ - onCancel?: () => void; - /** Callback when date-time changes */ - onChange?: (dateTime: DateTime | null) => void; - /** Placeholder text for the input field */ - placeholder?: string; - /** Whether to show the time selector */ - showTime?: boolean; - /** Whether to show the timezone selector */ - showTimeZone?: boolean; - /** Props for customizing the TimePicker component */ - timeSelectProps?: { - label?: string; - onChange?: (time: string) => void; - value?: null | string; - }; - /** Props for customizing the TimeZoneSelect component */ - timeZoneSelectProps?: { - label?: string; - onChange?: (timezone: string) => void; - value?: null | string; - }; - /** Initial or controlled dateTime value */ - value?: DateTime | null; -} - -export const DateTimePicker = ({ - dateCalendarProps = {}, - errorText = '', - format = 'yyyy-MM-dd HH:mm', - label = 'Select Date and Time', - onApply, - onCancel, - onChange, - placeholder = 'yyyy-MM-dd HH:mm', - showTime = true, - showTimeZone = true, - timeSelectProps = {}, - timeZoneSelectProps = {}, - value = null, -}: DateTimePickerProps) => { - const [anchorEl, setAnchorEl] = useState(null); - const [selectedDateTime, setSelectedDateTime] = useState( - value - ); - const [selectedTimeZone, setSelectedTimeZone] = useState( - timeZoneSelectProps.value || null - ); - - const TimePickerFieldProps: TextFieldProps = { - label: timeSelectProps?.label ?? 'Select Time', - noMarginTop: true, - }; - - useEffect(() => { - if (value) { - setSelectedDateTime(value); - } - }, [value]); - - useEffect(() => { - if (timeZoneSelectProps.value) { - setSelectedTimeZone(timeZoneSelectProps.value); - } - }, [timeZoneSelectProps.value]); - - const handleOpen = (event: React.MouseEvent) => { - setAnchorEl(event.currentTarget); - }; - - const handleClose = () => { - setAnchorEl(null); - if (onCancel) { - onCancel(); - } - }; - - const handleApply = () => { - setAnchorEl(null); - if (onChange) { - onChange(selectedDateTime); - } - if (onApply) { - onApply(); - } - }; - - const handleDateChange = (newDate: DateTime | null) => { - setSelectedDateTime((prev) => - newDate - ? newDate.set({ - hour: prev?.hour || 0, - minute: prev?.minute || 0, - }) - : null - ); - }; - - const handleTimeChange = (newTime: DateTime | null) => { - if (newTime) { - setSelectedDateTime((prev) => { - if (prev) { - // Ensure hour and minute are valid numbers - const newHour = newTime.hour; - const newMinute = newTime.minute; - - if (typeof newHour === 'number' && typeof newMinute === 'number') { - return prev.set({ hour: newHour, minute: newMinute }); - } - } - // Return the current `prev` value if newTime is invalid - return prev; - }); - } - if (timeSelectProps.onChange && newTime) { - timeSelectProps.onChange(newTime.toISOTime()); - } - }; - - const handleTimeZoneChange = (newTimeZone: string) => { - setSelectedTimeZone(newTimeZone); - if (timeZoneSelectProps.onChange) { - timeZoneSelectProps.onChange(newTimeZone); - } - }; - - return ( - - - - - - - ({ - '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { - justifyContent: 'space-between', - }, - '& .MuiDayCalendar-weekDayLabel': { - fontSize: '0.875rem', - }, - '& .MuiPickersCalendarHeader-label': { - fontFamily: theme.font.bold, - }, - '& .MuiPickersCalendarHeader-root': { - borderBottom: `1px solid ${theme.borderColors.divider}`, - fontSize: '0.875rem', - paddingBottom: theme.spacing(1), - }, - '& .MuiPickersDay-root': { - fontSize: '0.875rem', - margin: `${theme.spacing(0.5)}px`, - }, - borderRadius: `${theme.spacing(2)}`, - borderWidth: '0px', - })} - onChange={handleDateChange} - value={selectedDateTime} - {...dateCalendarProps} - /> - - {showTime && ( - - - - )} - {showTimeZone && ( - - - - )} - - - - - ({ - marginBottom: theme.spacing(1), - marginRight: theme.spacing(2), - })} - /> - - - - ); -}; diff --git a/packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx deleted file mode 100644 index 70ad6b2a39d..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimeRangePicker.stories.tsx +++ /dev/null @@ -1,77 +0,0 @@ -import { action } from '@storybook/addon-actions'; -import { DateTime } from 'luxon'; -import * as React from 'react'; - -import { DateTimeRangePicker } from './DateTimeRangePicker'; - -import type { Meta, StoryObj } from '@storybook/react'; - -type Story = StoryObj; - -export const Default: Story = { - args: { - endLabel: 'End Date and Time', - format: 'yyyy-MM-dd HH:mm', - onChange: action('DateTime range changed'), - startLabel: 'Start Date and Time', - }, - render: (args) => , -}; - -export const WithInitialValues: Story = { - render: () => { - const ComponentWithState: React.FC = () => { - const [start, setStart] = React.useState( - DateTime.now().minus({ days: 1 }) - ); - const [end, setEnd] = React.useState(DateTime.now()); - - const handleDateChange = ( - newStart: DateTime | null, - newEnd: DateTime | null - ) => { - setStart(newStart); - setEnd(newEnd); - action('DateTime range changed')(newStart?.toISO(), newEnd?.toISO()); - }; - - return ( - - ); - }; - - return ; - }, -}; - -const meta: Meta = { - argTypes: { - endLabel: { - control: 'text', - description: 'Label for the end date picker', - }, - format: { - control: 'text', - description: 'Format for displaying the date-time', - }, - onChange: { - action: 'DateTime range changed', - description: 'Callback when the date-time range changes', - }, - startLabel: { - control: 'text', - description: 'Label for the start date picker', - }, - }, - component: DateTimeRangePicker, - title: 'Components/DateTimeRangePicker', -}; - -export default meta; diff --git a/packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx deleted file mode 100644 index f3421b684c3..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ /dev/null @@ -1,59 +0,0 @@ -import { screen } from '@testing-library/react'; -import userEvent from '@testing-library/user-event'; -import * as React from 'react'; - -import { renderWithTheme } from 'src/utilities/testHelpers'; - -import { DateTimeRangePicker } from './DateTimeRangePicker'; - -describe('DateTimeRangePicker Component', () => { - const onChangeMock = vi.fn(); - - beforeEach(() => { - vi.clearAllMocks(); - }); - - it('should render start and end DateTimePickers with correct labels', () => { - renderWithTheme(); - - expect(screen.getByLabelText('Start Date')).toBeVisible(); - expect(screen.getByLabelText('End Date and Time')).toBeVisible(); - }); - - it('should call onChange when start date is changed', async () => { - renderWithTheme(); - - // Open start date picker - await userEvent.click(screen.getByLabelText('Start Date')); - - await userEvent.click(screen.getByRole('gridcell', { name: '10' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - // Check if the onChange function is called with the expected DateTime value - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ day: 10 }), - null - ); - }); - - it('should show error when end date-time is before start date-time', async () => { - renderWithTheme(); - - // Set start date-time to the 15th - const startDateField = screen.getByLabelText('Start Date'); - await userEvent.click(startDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '15' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - - // Open the end date picker - const endDateField = screen.getByLabelText('End Date and Time'); - await userEvent.click(endDateField); - - // Check if the date before the start date is disabled via a class or attribute - const invalidDate = screen.getByRole('gridcell', { name: '10' }); - expect(invalidDate).toHaveClass('Mui-disabled'); // or check for the specific attribute used - - // Confirm error message is not shown since the click was blocked - expect(screen.queryByText('Invalid date and Time')).not.toBeInTheDocument(); - }); -}); diff --git a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx deleted file mode 100644 index f4cb77b175a..00000000000 --- a/packages/ui/src/components/DatePicker/DateTimeRangePicker.tsx +++ /dev/null @@ -1,116 +0,0 @@ -import { Box } from '@linode/ui'; -import { DateTime } from 'luxon'; -import React, { useState } from 'react'; - -import { DateTimePicker } from './DateTimePicker'; - -interface DateTimeRangePickerProps { - /** Initial or controlled value for the end date-time */ - endDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - endLabel?: string; - /** Format for displaying the date-time */ - format?: string; - /** Callback when the date-time range changes */ - onChange?: ( - start: DateTime | null, - end: DateTime | null, - startTimeZone?: null | string - ) => void; - /** Whether to show the timezone field for the end date picker */ - showEndTimeZone?: boolean; - /** Initial or controlled value for the start date-time */ - startDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - startLabel?: string; - /** Initial or controlled value for the start timezone */ - startTimeZoneValue?: null | string; -} - -export const DateTimeRangePicker = ({ - endDateTimeValue = null, - endLabel = 'End Date and Time', - format = 'yyyy-MM-dd HH:mm', - onChange, - showEndTimeZone = false, - startDateTimeValue = null, - startLabel = 'Start Date and Time', - startTimeZoneValue = null, -}: DateTimeRangePickerProps) => { - const [startDateTime, setStartDateTime] = useState( - startDateTimeValue - ); - const [endDateTime, setEndDateTime] = useState( - endDateTimeValue - ); - const [startTimeZone, setStartTimeZone] = useState( - startTimeZoneValue - ); - const [error, setError] = useState(); - - const handleStartDateTimeChange = (newStart: DateTime | null) => { - setStartDateTime(newStart); - - if (endDateTime && newStart && endDateTime >= newStart) { - setError(undefined); // Clear error if valid - } - - if (onChange) { - onChange(newStart, endDateTime, startTimeZone); - } - }; - - const handleEndDateTimeChange = (newEnd: DateTime | null) => { - if (startDateTime && newEnd && newEnd < startDateTime) { - setError('End date/time must be after the start date/time.'); - } else { - setEndDateTime(newEnd); - setError(undefined); - } - - if (onChange) { - onChange(startDateTime, newEnd, startTimeZone); - } - }; - - const handleStartTimeZoneChange = (newTimeZone: null | string) => { - setStartTimeZone(newTimeZone); - - if (onChange) { - onChange(startDateTime, endDateTime, newTimeZone); - } - }; - - return ( - - {/* Start DateTime Picker */} - - - {/* End DateTime Picker */} - - - ); -}; diff --git a/packages/ui/src/components/DatePicker/TimeZoneSelect.tsx b/packages/ui/src/components/DatePicker/TimeZoneSelect.tsx deleted file mode 100644 index 19312098007..00000000000 --- a/packages/ui/src/components/DatePicker/TimeZoneSelect.tsx +++ /dev/null @@ -1,64 +0,0 @@ -import { DateTime } from 'luxon'; -import React from 'react'; - -import { timezones } from 'src/assets/timezones/timezones'; -import { Autocomplete } from 'src/components/Autocomplete/Autocomplete'; - -type Timezone = typeof timezones[number]; - -interface TimeZoneSelectProps { - disabled?: boolean; - errorText?: string; - label?: string; - noMarginTop?: boolean; - onChange: (timezone: string) => void; - value: null | string; -} - -const getOptionLabel = ({ label, offset }: Timezone) => { - const minutes = (Math.abs(offset) % 60).toLocaleString(undefined, { - minimumIntegerDigits: 2, - useGrouping: false, - }); - const hours = Math.floor(Math.abs(offset) / 60); - const isPositive = Math.abs(offset) === offset ? '+' : '-'; - - return `(GMT ${isPositive}${hours}:${minutes}) ${label}`; -}; - -const getTimezoneOptions = () => { - return timezones - .map((tz) => { - const offset = DateTime.now().setZone(tz.name).offset; - const label = getOptionLabel({ ...tz, offset }); - return { label, offset, value: tz.name }; - }) - .sort((a, b) => a.offset - b.offset); -}; - -const timezoneOptions = getTimezoneOptions(); - -export const TimeZoneSelect = ({ - disabled = false, - errorText, - label = 'Timezone', - noMarginTop = false, - onChange, - value, -}: TimeZoneSelectProps) => { - return ( - option.value === value) ?? undefined - } - autoHighlight - disabled={disabled} - errorText={errorText} - label={label} - noMarginTop={noMarginTop} - onChange={(e, option) => onChange(option?.value || '')} - options={timezoneOptions} - placeholder="Choose a Timezone" - /> - ); -}; diff --git a/packages/ui/src/components/DatePicker/index.ts b/packages/ui/src/components/DatePicker/index.ts deleted file mode 100644 index a48b62e4dfc..00000000000 --- a/packages/ui/src/components/DatePicker/index.ts +++ /dev/null @@ -1 +0,0 @@ -export * from './DatePicker'; From b8f2d739124874ee61e827beb3a7fa7c5f6347d9 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 4 Dec 2024 11:51:16 -0600 Subject: [PATCH 26/45] Adjust broken tests --- .../src/components/DatePicker/DateTimeRangePicker.test.tsx | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index f3421b684c3..c1303d84437 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -16,7 +16,7 @@ describe('DateTimeRangePicker Component', () => { it('should render start and end DateTimePickers with correct labels', () => { renderWithTheme(); - expect(screen.getByLabelText('Start Date')).toBeVisible(); + expect(screen.getByLabelText('Start Date and Time')).toBeVisible(); expect(screen.getByLabelText('End Date and Time')).toBeVisible(); }); @@ -24,7 +24,7 @@ describe('DateTimeRangePicker Component', () => { renderWithTheme(); // Open start date picker - await userEvent.click(screen.getByLabelText('Start Date')); + await userEvent.click(screen.getByLabelText('Start Date and Time')); await userEvent.click(screen.getByRole('gridcell', { name: '10' })); await userEvent.click(screen.getByRole('button', { name: 'Apply' })); @@ -32,6 +32,7 @@ describe('DateTimeRangePicker Component', () => { // Check if the onChange function is called with the expected DateTime value expect(onChangeMock).toHaveBeenCalledWith( expect.objectContaining({ day: 10 }), + null, null ); }); @@ -40,7 +41,7 @@ describe('DateTimeRangePicker Component', () => { renderWithTheme(); // Set start date-time to the 15th - const startDateField = screen.getByLabelText('Start Date'); + const startDateField = screen.getByLabelText('Start Date and Time'); await userEvent.click(startDateField); await userEvent.click(screen.getByRole('gridcell', { name: '15' })); await userEvent.click(screen.getByRole('button', { name: 'Apply' })); From e52304339f8884bbbc4d3ea21ea2fd06554b0040 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 4 Dec 2024 11:58:14 -0600 Subject: [PATCH 27/45] Update TimeZoneSelect.tsx --- packages/manager/src/components/DatePicker/TimeZoneSelect.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx index 19312098007..f4bd68c97a3 100644 --- a/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx +++ b/packages/manager/src/components/DatePicker/TimeZoneSelect.tsx @@ -1,8 +1,8 @@ +import { Autocomplete } from '@linode/ui'; import { DateTime } from 'luxon'; import React from 'react'; import { timezones } from 'src/assets/timezones/timezones'; -import { Autocomplete } from 'src/components/Autocomplete/Autocomplete'; type Timezone = typeof timezones[number]; From f0dd5c09520d39c3c8b5e802ef1e41a75e63305c Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 9 Dec 2024 11:50:57 -0600 Subject: [PATCH 28/45] Code cleanup --- .../DatePicker/DatePicker.stories.tsx | 7 +- .../DatePicker/DateTimePicker.stories.tsx | 5 +- .../components/DatePicker/DateTimePicker.tsx | 77 +++++++------------ .../DatePicker/DateTimeRangePicker.tsx | 6 +- 4 files changed, 40 insertions(+), 55 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DatePicker.stories.tsx b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx index db3925b03eb..6e9d4408783 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.stories.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.stories.tsx @@ -1,19 +1,17 @@ import { action } from '@storybook/addon-actions'; -import { DateTime } from 'luxon'; import * as React from 'react'; import { DatePicker } from './DatePicker'; import type { Meta, StoryObj } from '@storybook/react'; +import type { DateTime } from 'luxon'; type Story = StoryObj; export const ControlledExample: Story = { render: () => { const ControlledDatePicker = () => { - const [selectedDate, setSelectedDate] = React.useState( - DateTime.now() - ); + const [selectedDate, setSelectedDate] = React.useState(); const handleChange = (newDate: DateTime | null) => { setSelectedDate(newDate); @@ -26,6 +24,7 @@ export const ControlledExample: Story = { helperText="This is a controlled DatePicker" label="Controlled Date Picker" onChange={handleChange} + placeholder="yyyy-MM-dd" value={selectedDate} /> ); diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx index 32d65dfa0b7..bb3f0c48233 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.stories.tsx @@ -1,10 +1,10 @@ import { action } from '@storybook/addon-actions'; -import { DateTime } from 'luxon'; import * as React from 'react'; import { DateTimePicker } from './DateTimePicker'; import type { Meta, StoryObj } from '@storybook/react'; +import type { DateTime } from 'luxon'; type Story = StoryObj; @@ -14,7 +14,7 @@ export const ControlledExample: Story = { const [ selectedDateTime, setSelectedDateTime, - ] = React.useState(DateTime.now()); + ] = React.useState(null); // Start with null const handleChange = (newDateTime: DateTime | null) => { setSelectedDateTime(newDateTime); @@ -32,6 +32,7 @@ export const ControlledExample: Story = { onCancel={action('Cancel clicked')} onChange={handleChange} placeholder="yyyy-MM-dd HH:mm" + showTimeZone={false} timeSelectProps={{ label: 'Select Time' }} value={selectedDateTime} /> diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 027b25822d2..a67b5ccb155 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -82,39 +82,6 @@ export const DateTimePicker = ({ noMarginTop: true, }; - useEffect(() => { - if (value) { - setSelectedDateTime(value); - } - }, [value]); - - useEffect(() => { - if (timeZoneSelectProps.value) { - setSelectedTimeZone(timeZoneSelectProps.value); - } - }, [timeZoneSelectProps.value]); - - const handleOpen = (event: React.MouseEvent) => { - setAnchorEl(event.currentTarget); - }; - - const handleClose = () => { - setAnchorEl(null); - if (onCancel) { - onCancel(); - } - }; - - const handleApply = () => { - setAnchorEl(null); - if (onChange) { - onChange(selectedDateTime); - } - if (onApply) { - onApply(); - } - }; - const handleDateChange = (newDate: DateTime | null) => { setSelectedDateTime((prev) => newDate @@ -130,7 +97,6 @@ export const DateTimePicker = ({ if (newTime) { setSelectedDateTime((prev) => { if (prev) { - // Ensure hour and minute are valid numbers const newHour = newTime.hour; const newMinute = newTime.minute; @@ -138,13 +104,9 @@ export const DateTimePicker = ({ return prev.set({ hour: newHour, minute: newMinute }); } } - // Return the current `prev` value if newTime is invalid return prev; }); } - if (timeSelectProps.onChange && newTime) { - timeSelectProps.onChange(newTime.toISOTime()); - } }; const handleTimeZoneChange = (newTimeZone: string) => { @@ -154,6 +116,29 @@ export const DateTimePicker = ({ } }; + const handleApply = () => { + setAnchorEl(null); + if (onChange) { + onChange(selectedDateTime); + } + if (onApply) { + onApply(); + } + }; + + const handleClose = () => { + setAnchorEl(null); + if (onCancel) { + onCancel(); + } + }; + + useEffect(() => { + if (timeZoneSelectProps.value) { + setSelectedTimeZone(timeZoneSelectProps.value); + } + }, [timeZoneSelectProps.value]); + return ( @@ -168,7 +153,7 @@ export const DateTimePicker = ({ InputProps={{ readOnly: true }} errorText={errorText} label={label} - onClick={handleOpen} + onClick={(event) => setAnchorEl(event.currentTarget)} placeholder={placeholder} /> @@ -181,6 +166,9 @@ export const DateTimePicker = ({ > ({ '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { justifyContent: 'space-between', @@ -203,9 +191,6 @@ export const DateTimePicker = ({ borderRadius: `${theme.spacing(2)}`, borderWidth: '0px', })} - onChange={handleDateChange} - value={selectedDateTime} - {...dateCalendarProps} /> )} diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index f4cb77b175a..0d973178715 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -17,8 +17,10 @@ interface DateTimeRangePickerProps { end: DateTime | null, startTimeZone?: null | string ) => void; - /** Whether to show the timezone field for the end date picker */ + /** Whether to show the end timezone field for the end date picker */ showEndTimeZone?: boolean; + /** Whether to show the start timezone field for the end date picker */ + showStartTimeZone?: boolean; /** Initial or controlled value for the start date-time */ startDateTimeValue?: DateTime | null; /** Custom labels for the start and end date/time fields */ @@ -33,6 +35,7 @@ export const DateTimeRangePicker = ({ format = 'yyyy-MM-dd HH:mm', onChange, showEndTimeZone = false, + showStartTimeZone = false, startDateTimeValue = null, startLabel = 'Start Date and Time', startTimeZoneValue = null, @@ -94,6 +97,7 @@ export const DateTimeRangePicker = ({ format={format} label={startLabel} onChange={handleStartDateTimeChange} + showTimeZone={showStartTimeZone} timeSelectProps={{ label: 'Start Time' }} value={startDateTime} /> From 9abec60802f44ab47f9d5e972437a87548257639 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 9 Dec 2024 14:16:07 -0600 Subject: [PATCH 29/45] Add validation for start date agains end date. --- .../DatePicker/DateTimeRangePicker.test.tsx | 53 +++++++++++++++++++ .../DatePicker/DateTimeRangePicker.tsx | 42 ++++++++++----- 2 files changed, 83 insertions(+), 12 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index c1303d84437..568e9559e06 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -57,4 +57,57 @@ describe('DateTimeRangePicker Component', () => { // Confirm error message is not shown since the click was blocked expect(screen.queryByText('Invalid date and Time')).not.toBeInTheDocument(); }); + + it('should show error when start date-time is after end date-time', async () => { + renderWithTheme( + + ); + + // Set the end date-time to the 15th + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Set the start date-time to the 10th (which is earlier than the end date-time) + const startDateField = screen.getByLabelText('Start Date and Time'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '20' })); // Invalid date + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Confirm the error message is displayed + expect( + screen.getByText('Start date/time cannot be after the end date/time.') + ).toBeInTheDocument(); + }); + + it('should display custom error messages when start date-time is after end date-time', async () => { + renderWithTheme( + + ); + + // Set the end date-time to the 15th + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Set the start date-time to the 20th (which is after the end date-time) + const startDateField = screen.getByLabelText('Start Date and Time'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '20' })); // Invalid date + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Confirm the custom error message is displayed for the start date + expect(screen.getByText('Custom start date error')).toBeInTheDocument(); + }); }); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 0d973178715..d5fa4989b71 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,10 +1,13 @@ import { Box } from '@linode/ui'; -import { DateTime } from 'luxon'; import React, { useState } from 'react'; import { DateTimePicker } from './DateTimePicker'; +import type { DateTime } from 'luxon'; + interface DateTimeRangePickerProps { + /** Custom error message for invalid end date */ + endDateErrorMessage?: string; /** Initial or controlled value for the end date-time */ endDateTimeValue?: DateTime | null; /** Custom labels for the start and end date/time fields */ @@ -21,6 +24,8 @@ interface DateTimeRangePickerProps { showEndTimeZone?: boolean; /** Whether to show the start timezone field for the end date picker */ showStartTimeZone?: boolean; + /** Custom error message for invalid start date */ + startDateErrorMessage?: string; /** Initial or controlled value for the start date-time */ startDateTimeValue?: DateTime | null; /** Custom labels for the start and end date/time fields */ @@ -30,12 +35,14 @@ interface DateTimeRangePickerProps { } export const DateTimeRangePicker = ({ + endDateErrorMessage = 'End date/time cannot be before the start date/time.', endDateTimeValue = null, endLabel = 'End Date and Time', format = 'yyyy-MM-dd HH:mm', onChange, showEndTimeZone = false, showStartTimeZone = false, + startDateErrorMessage = 'Start date/time cannot be after the end date/time.', startDateTimeValue = null, startLabel = 'Start Date and Time', startTimeZoneValue = null, @@ -51,12 +58,27 @@ export const DateTimeRangePicker = ({ ); const [error, setError] = useState(); + const validateDates = ( + start: DateTime | null, + end: DateTime | null, + source: 'end' | 'start' + ) => { + if (start && end) { + if (source === 'start' && start > end) { + setError(startDateErrorMessage); + return; + } + if (source === 'end' && end < start) { + setError(endDateErrorMessage); + return; + } + } + setError(undefined); // Clear error if valid + }; + const handleStartDateTimeChange = (newStart: DateTime | null) => { setStartDateTime(newStart); - - if (endDateTime && newStart && endDateTime >= newStart) { - setError(undefined); // Clear error if valid - } + validateDates(newStart, endDateTime, 'start'); if (onChange) { onChange(newStart, endDateTime, startTimeZone); @@ -64,12 +86,8 @@ export const DateTimeRangePicker = ({ }; const handleEndDateTimeChange = (newEnd: DateTime | null) => { - if (startDateTime && newEnd && newEnd < startDateTime) { - setError('End date/time must be after the start date/time.'); - } else { - setEndDateTime(newEnd); - setError(undefined); - } + setEndDateTime(newEnd); + validateDates(startDateTime, newEnd, 'end'); if (onChange) { onChange(startDateTime, newEnd, startTimeZone); @@ -107,7 +125,7 @@ export const DateTimeRangePicker = ({ timeZoneSelectProps={{ value: startTimeZone, // Automatically reflect the start timezone }} - dateCalendarProps={{ minDate: startDateTime ?? DateTime.now() }} + dateCalendarProps={{ minDate: startDateTime ?? undefined }} format={format} label={endLabel} onChange={handleEndDateTimeChange} From fa3e1861c5b2888ef496027503fdf6aaff1a8c09 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 10 Dec 2024 08:42:00 -0600 Subject: [PATCH 30/45] Adjust styles for TimePicker component. --- .../src/components/DatePicker/DatePicker.tsx | 1 + .../components/DatePicker/DateTimePicker.tsx | 28 ++++++++++++++++++- 2 files changed, 28 insertions(+), 1 deletion(-) diff --git a/packages/manager/src/components/DatePicker/DatePicker.tsx b/packages/manager/src/components/DatePicker/DatePicker.tsx index a3a5a722e82..f8fc1965ac8 100644 --- a/packages/manager/src/components/DatePicker/DatePicker.tsx +++ b/packages/manager/src/components/DatePicker/DatePicker.tsx @@ -56,6 +56,7 @@ export const DatePicker = ({ value={value} {...props} slotProps={{ + // TODO: Move styling customization to global theme styles. popper: { sx: { '& .MuiDayCalendar-weekDayLabel': { diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index a67b5ccb155..0bbe09e5f21 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -169,6 +169,7 @@ export const DateTimePicker = ({ onChange={handleDateChange} value={selectedDateTime || null} {...dateCalendarProps} + // TODO: Move styling customization to global theme styles. sx={(theme: Theme) => ({ '& .MuiDayCalendar-weekContainer, & .MuiDayCalendar-header': { justifyContent: 'space-between', @@ -201,7 +202,32 @@ export const DateTimePicker = ({ ({ + justifyContent: 'center', + marginBottom: theme.spacing(1 / 2), + marginTop: theme.spacing(1 / 2), + padding: 0, + }), + }, + layout: { + sx: (theme: Theme) => ({ + '& .MuiPickersLayout-contentWrapper': { + borderBottom: `1px solid ${theme.borderColors.divider}`, + }, + border: `1px solid ${theme.borderColors.divider}`, + }), + }, + openPickerButton: { + sx: { padding: 0 }, + }, + popper: { + sx: (theme: Theme) => ({ + ul: { + borderColor: `${theme.borderColors.divider} !important`, + }, + }), + }, textField: TimePickerFieldProps, }} onChange={handleTimeChange} From 006371ec5a57ea763ce133533eb3a9899a686227 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 10 Dec 2024 12:33:12 -0600 Subject: [PATCH 31/45] Add the functionality to support Date Presets --- .../DatePicker/DateTimeRangePicker.tsx | 152 ++++++++++++++---- 1 file changed, 118 insertions(+), 34 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index d5fa4989b71..9ef6af0dc43 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,11 +1,12 @@ -import { Box } from '@linode/ui'; +import { Autocomplete, Box, Button } from '@linode/ui'; +import { DateTime } from 'luxon'; import React, { useState } from 'react'; import { DateTimePicker } from './DateTimePicker'; -import type { DateTime } from 'luxon'; - interface DateTimeRangePickerProps { + /** If true, shows the date presets field instead of the date pickers */ + enablePresets?: boolean; /** Custom error message for invalid end date */ endDateErrorMessage?: string; /** Initial or controlled value for the end date-time */ @@ -22,7 +23,7 @@ interface DateTimeRangePickerProps { ) => void; /** Whether to show the end timezone field for the end date picker */ showEndTimeZone?: boolean; - /** Whether to show the start timezone field for the end date picker */ + /** Whether to show the start timezone field for the start date picker */ showStartTimeZone?: boolean; /** Custom error message for invalid start date */ startDateErrorMessage?: string; @@ -35,6 +36,7 @@ interface DateTimeRangePickerProps { } export const DateTimeRangePicker = ({ + enablePresets = false, endDateErrorMessage = 'End date/time cannot be before the start date/time.', endDateTimeValue = null, endLabel = 'End Date and Time', @@ -57,6 +59,7 @@ export const DateTimeRangePicker = ({ startTimeZoneValue ); const [error, setError] = useState(); + const [showPresets, setShowPresets] = useState(enablePresets); const validateDates = ( start: DateTime | null, @@ -102,37 +105,118 @@ export const DateTimeRangePicker = ({ } }; + const handlePresetSelection = (value: string) => { + const now = DateTime.now(); + let newStartDateTime: DateTime | null = null; + let newEndDateTime: DateTime | null = null; + + switch (value) { + case '24hours': + newStartDateTime = now.minus({ hours: 24 }); + newEndDateTime = now; + break; + case '7days': + newStartDateTime = now.minus({ days: 7 }); + newEndDateTime = now; + break; + case '30days': + newStartDateTime = now.minus({ days: 30 }); + newEndDateTime = now; + break; + case 'this_month': + newStartDateTime = now.startOf('month'); + newEndDateTime = now.endOf('month'); + break; + case 'last_month': + const lastMonth = now.minus({ months: 1 }); + newStartDateTime = lastMonth.startOf('month'); + newEndDateTime = lastMonth.endOf('month'); + break; + case 'custom_range': + newStartDateTime = null; + newEndDateTime = null; + break; + default: + return; + } + + setStartDateTime(newStartDateTime); + setEndDateTime(newEndDateTime); + + if (onChange) { + onChange(newStartDateTime, newEndDateTime, startTimeZone); + } + + // Show date pickers after selecting a preset + if (value !== 'custom_range') { + setShowPresets(false); + } + }; + return ( - - {/* Start DateTime Picker */} - - - {/* End DateTime Picker */} - + + {showPresets ? ( + { + if (selection) { + handlePresetSelection(selection.value); + } + }} + options={[ + { label: 'Last 24 Hours', value: '24hours' }, + { label: 'Last 7 Days', value: '7days' }, + { label: 'Last 30 Days', value: '30days' }, + { label: 'This Month', value: 'this_month' }, + { label: 'Last Month', value: 'last_month' }, + { label: 'Custom Range', value: 'custom_range' }, + ]} + value={{ + label: 'Select a preset', + value: '', + }} + fullWidth + label="Date Presets" + placeholder="Select Date" + /> + ) : ( + <> + + + + + + + )} ); }; From 8c465b2378f1d8a69914a5994c639afe33bb2f9f Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 16 Dec 2024 14:30:53 -0600 Subject: [PATCH 32/45] Update presets functionality and add test coverage. --- .../DatePicker/DateTimeRangePicker.test.tsx | 321 +++++++++++++----- .../DatePicker/DateTimeRangePicker.tsx | 240 +++---------- 2 files changed, 296 insertions(+), 265 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index 4b5a5b75f33..83f9191d92c 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -1,5 +1,6 @@ import { screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; +import { DateTime } from 'luxon'; import * as React from 'react'; import { renderWithTheme } from 'src/utilities/testHelpers'; @@ -98,121 +99,287 @@ describe('DateTimeRangePicker Component', () => { /> ); + // Set the end date-time to the 15th + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Set the start date-time to the 20th (which is earlier than the end date-time) + const startDateField = screen.getByLabelText('Start Date and Time'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '20' })); // Invalid date + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Confirm the custom error message is displayed for the start date expect(screen.getByText('Custom start date error')).toBeInTheDocument(); - expect(screen.getByText('Custom end date error')).toBeInTheDocument(); }); -}); -import { screen } from '@testing-library/react'; -import userEvent from '@testing-library/user-event'; -import * as React from 'react'; -import { renderWithTheme } from 'src/utilities/testHelpers'; + it('should set the date range for the last 24 hours when the "Last 24 Hours" preset is selected', async () => { + renderWithTheme( + + ); -import { DateTimeRangePicker } from './DateTimeRangePicker'; + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); -describe('DateTimeRangePicker Component', () => { - const onChangeMock = vi.fn(); + // Select the "Last 24 Hours" option + const last24HoursOption = screen.getByText('Last 24 Hours'); + await userEvent.click(last24HoursOption); - beforeEach(() => { - vi.clearAllMocks(); + // Verify that onChange is called with the correct date range + const now = DateTime.now(); + const expectedStartDateTime = now.minus({ hours: 24 }); + + const expectedEndDateTime = now; + + const expectedStartDateValue = expectedStartDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + const expectedEndDateValue = expectedEndDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ + day: expectedStartDateTime.day, + month: expectedStartDateTime.month, + year: expectedStartDateTime.year, + }), + expect.objectContaining({ + day: expectedEndDateTime.day, + month: expectedEndDateTime.month, + year: expectedEndDateTime.year, + }), + null + ); + + // Verify that Date input fields has the correct date range + expect( + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(expectedStartDateValue); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(expectedEndDateValue); }); - it('should render start and end DateTimePickers with correct labels', () => { - renderWithTheme(); + it('should set the date range for the last 7 days when the "Last 7 Days" preset is selected', async () => { + renderWithTheme( + + ); - expect(screen.getByLabelText('Start Date and Time')).toBeVisible(); - expect(screen.getByLabelText('End Date and Time')).toBeVisible(); + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "Last 7 Days" option + const last7DaysOption = screen.getByText('Last 7 Days'); + await userEvent.click(last7DaysOption); + + // Verify that onChange is called with the correct date range + const now = DateTime.now(); + const expectedStartDateTime = now.minus({ days: 7 }); + const expectedEndDateTime = now; + + const expectedStartDateValue = expectedStartDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + const expectedEndDateValue = expectedEndDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ + day: expectedStartDateTime.day, + month: expectedStartDateTime.month, + year: expectedStartDateTime.year, + }), + expect.objectContaining({ + day: expectedEndDateTime.day, + month: expectedEndDateTime.month, + year: expectedEndDateTime.year, + }), + null + ); + + // Verify that Date input fields have the correct date range + expect( + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(expectedStartDateValue); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(expectedEndDateValue); }); - it('should call onChange when start date is changed', async () => { - renderWithTheme(); + it('should set the date range for the last 30 days when the "Last 30 Days" preset is selected', async () => { + renderWithTheme( + + ); - // Open start date picker - await userEvent.click(screen.getByLabelText('Start Date and Time')); + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); - await userEvent.click(screen.getByRole('gridcell', { name: '10' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Select the "Last 30 Days" option + const last30DaysOption = screen.getByText('Last 30 Days'); + await userEvent.click(last30DaysOption); - // Check if the onChange function is called with the expected DateTime value + const now = DateTime.now(); + const expectedStartDateTime = now.minus({ days: 30 }); + const expectedEndDateTime = now; + + // Use the same format as the component for verification + const expectedStartDateValue = expectedStartDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + const expectedEndDateValue = expectedEndDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + + // Verify that onChange is called with the correct date range expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ day: 10 }), - null, + expect.objectContaining({ + day: expectedStartDateTime.day, + month: expectedStartDateTime.month, + year: expectedStartDateTime.year, + }), + expect.objectContaining({ + day: expectedEndDateTime.day, + month: expectedEndDateTime.month, + year: expectedEndDateTime.year, + }), null ); + + // Verify the input fields display the correct values + expect( + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(expectedStartDateValue); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(expectedEndDateValue); }); - it('should show error when end date-time is before start date-time', async () => { - renderWithTheme(); + it('should set the date range for this month when the "This Month" preset is selected', async () => { + renderWithTheme( + + ); - // Set start date-time to the 15th - const startDateField = screen.getByLabelText('Start Date and Time'); - await userEvent.click(startDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '15' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); - // Open the end date picker - const endDateField = screen.getByLabelText('End Date and Time'); - await userEvent.click(endDateField); + // Select the "This Month" option + const thisMonthOption = screen.getByText('This Month'); + await userEvent.click(thisMonthOption); - // Check if the date before the start date is disabled via a class or attribute - const invalidDate = screen.getByRole('gridcell', { name: '10' }); - expect(invalidDate).toHaveClass('Mui-disabled'); // or check for the specific attribute used + const now = DateTime.now(); + const expectedStartDateTime = now.startOf('month'); + const expectedEndDateTime = now.endOf('month'); - // Confirm error message is not shown since the click was blocked - expect(screen.queryByText('Invalid date and Time')).not.toBeInTheDocument(); + // Use the same format as the component for verification + const expectedStartDateValue = expectedStartDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + const expectedEndDateValue = expectedEndDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + + // Verify that onChange is called with the correct date range + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ + day: expectedStartDateTime.day, + month: expectedStartDateTime.month, + year: expectedStartDateTime.year, + }), + expect.objectContaining({ + day: expectedEndDateTime.day, + month: expectedEndDateTime.month, + year: expectedEndDateTime.year, + }), + null + ); + + // Verify the input fields display the correct values + expect( + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(expectedStartDateValue); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(expectedEndDateValue); }); - it('should show error when start date-time is after end date-time', async () => { + it('should set the date range for last month when the "Last Month" preset is selected', async () => { renderWithTheme( - + ); - // Set the end date-time to the 15th - const endDateField = screen.getByLabelText('End Date and Time'); - await userEvent.click(endDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '15' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); - // Set the start date-time to the 10th (which is earlier than the end date-time) - const startDateField = screen.getByLabelText('Start Date and Time'); - await userEvent.click(startDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '20' })); // Invalid date - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Select the "Last Month" option + const lastMonthOption = screen.getByText('Last Month'); + await userEvent.click(lastMonthOption); - // Confirm the error message is displayed + const now = DateTime.now(); + const lastMonth = now.minus({ months: 1 }); + const expectedStartDateTime = lastMonth.startOf('month'); + const expectedEndDateTime = lastMonth.endOf('month'); + + // Use the same format as the component for verification + const expectedStartDateValue = expectedStartDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + const expectedEndDateValue = expectedEndDateTime.toFormat( + 'yyyy-MM-dd HH:mm' + ); + + // Verify that onChange is called with the correct date range + expect(onChangeMock).toHaveBeenCalledWith( + expect.objectContaining({ + day: expectedStartDateTime.day, + month: expectedStartDateTime.month, + year: expectedStartDateTime.year, + }), + expect.objectContaining({ + day: expectedEndDateTime.day, + month: expectedEndDateTime.month, + year: expectedEndDateTime.year, + }), + null + ); + + // Verify the input fields display the correct values expect( - screen.getByText('Start date/time cannot be after the end date/time.') - ).toBeInTheDocument(); + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(expectedStartDateValue); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(expectedEndDateValue); }); - it('should display custom error messages when start date-time is after end date-time', async () => { + it('should display the date range fields with empty values when the "Custom Range" preset is selected', async () => { renderWithTheme( - + ); - // Set the end date-time to the 15th - const endDateField = screen.getByLabelText('End Date and Time'); - await userEvent.click(endDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '15' })); - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); - // Set the start date-time to the 20th (which is after the end date-time) - const startDateField = screen.getByLabelText('Start Date and Time'); - await userEvent.click(startDateField); - await userEvent.click(screen.getByRole('gridcell', { name: '20' })); // Invalid date - await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + // Select the "Custom Range" option + const customRange = screen.getByText('Custom Range'); + await userEvent.click(customRange); - // Confirm the custom error message is displayed for the start date - expect(screen.getByText('Custom start date error')).toBeInTheDocument(); + // Verify the input fields display the correct values + expect( + screen.getByRole('textbox', { name: 'Start Date and Time' }) + ).toHaveValue(''); + expect( + screen.getByRole('textbox', { name: 'End Date and Time' }) + ).toHaveValue(''); + expect(screen.getByRole('button', { name: 'Presets' })).toBeInTheDocument(); }); }); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 77816eb221c..0013df716a4 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,152 +1,10 @@ -import { Box } from '@linode/ui'; +import { Autocomplete, Box, StyledActionButton } from '@linode/ui'; +import { DateTime } from 'luxon'; import React, { useState } from 'react'; import { DateTimePicker } from './DateTimePicker'; import type { SxProps, Theme } from '@mui/material/styles'; -import type { DateTime } from 'luxon'; - -interface DateTimeRangePickerProps { - /** Custom error message for invalid end date */ - endDateErrorMessage?: string; - /** Initial or controlled value for the end date-time */ - endDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - endLabel?: string; - /** Format for displaying the date-time */ - format?: string; - /** Callback when the date-time range changes */ - onChange: ( - start: DateTime | null, - end: DateTime | null, - startTimeZone?: null | string - ) => void; - /** Whether to show the end timezone field for the end date picker */ - showEndTimeZone?: boolean; - /** Whether to show the start timezone field for the end date picker */ - showStartTimeZone?: boolean; - /** Custom error message for invalid start date */ - startDateErrorMessage?: string; - /** Initial or controlled value for the start date-time */ - startDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - startLabel?: string; - /** Initial or controlled value for the start timezone */ - startTimeZoneValue?: null | string; - /** - * Any additional styles to apply to the root element. - */ - sx?: SxProps; -} - -export const DateTimeRangePicker = ({ - endDateErrorMessage, - endDateTimeValue = null, - endLabel = 'End Date and Time', - format = 'yyyy-MM-dd HH:mm', - onChange, - showEndTimeZone = false, - showStartTimeZone = false, - startDateErrorMessage, - startDateTimeValue = null, - startLabel = 'Start Date and Time', - startTimeZoneValue = null, - sx, -}: DateTimeRangePickerProps) => { - const [startDateTime, setStartDateTime] = useState( - startDateTimeValue - ); - const [endDateTime, setEndDateTime] = useState( - endDateTimeValue - ); - const [startTimeZone, setStartTimeZone] = useState( - startTimeZoneValue - ); - - const [startDateError, setStartDateError] = useState(); - const [endDateError, setEndDateError] = useState(); - - const validateDates = ( - start: DateTime | null, - end: DateTime | null, - source: 'end' | 'start' - ) => { - if (start && end) { - if (source === 'start' && start > end) { - setStartDateError('Start date/time cannot be after the end date/time.'); - setEndDateError(undefined); - return; - } - if (source === 'end' && end < start) { - setEndDateError('End date/time cannot be before the start date/time.'); - setStartDateError(undefined); - return; - } - } - // Reset validation errors if valid - setStartDateError(undefined); - setEndDateError(undefined); - }; - - const handleStartDateTimeChange = (newStart: DateTime | null) => { - setStartDateTime(newStart); - validateDates(newStart, endDateTime, 'start'); - - onChange(newStart, endDateTime, startTimeZone); - }; - - const handleEndDateTimeChange = (newEnd: DateTime | null) => { - setEndDateTime(newEnd); - validateDates(startDateTime, newEnd, 'end'); - - onChange(startDateTime, newEnd, startTimeZone); - }; - - const handleStartTimeZoneChange = (newTimeZone: null | string) => { - setStartTimeZone(newTimeZone); - - onChange(startDateTime, endDateTime, newTimeZone); - }; - - return ( - - {/* Start DateTime Picker */} - - - {/* End DateTime Picker */} - - - ); -}; -import { Autocomplete, Box, Button } from '@linode/ui'; -import { DateTime } from 'luxon'; -import React, { useState } from 'react'; - -import { DateTimePicker } from './DateTimePicker'; interface DateTimeRangePickerProps { /** If true, shows the date presets field instead of the date pickers */ @@ -177,6 +35,10 @@ interface DateTimeRangePickerProps { startLabel?: string; /** Initial or controlled value for the start timezone */ startTimeZoneValue?: null | string; + /** + * Any additional styles to apply to the root element. + */ + sx?: SxProps; } export const DateTimeRangePicker = ({ @@ -192,6 +54,7 @@ export const DateTimeRangePicker = ({ startDateTimeValue = null, startLabel = 'Start Date and Time', startTimeZoneValue = null, + sx, }: DateTimeRangePickerProps) => { const [startDateTime, setStartDateTime] = useState( startDateTimeValue @@ -202,7 +65,8 @@ export const DateTimeRangePicker = ({ const [startTimeZone, setStartTimeZone] = useState( startTimeZoneValue ); - const [error, setError] = useState(); + const [startDateError, setStartDateError] = useState(null); + const [endDateError, setEndDateError] = useState(null); const [showPresets, setShowPresets] = useState(enablePresets); const validateDates = ( @@ -212,15 +76,17 @@ export const DateTimeRangePicker = ({ ) => { if (start && end) { if (source === 'start' && start > end) { - setError(startDateErrorMessage); + setStartDateError(startDateErrorMessage); return; } if (source === 'end' && end < start) { - setError(endDateErrorMessage); + setEndDateError(endDateErrorMessage); return; } } - setError(undefined); // Clear error if valid + // Reset validation errors if valid + setStartDateError(null); + setEndDateError(null); }; const handleStartDateTimeChange = (newStart: DateTime | null) => { @@ -292,13 +158,11 @@ export const DateTimeRangePicker = ({ } // Show date pickers after selecting a preset - if (value !== 'custom_range') { - setShowPresets(false); - } + setShowPresets(false); }; return ( - + {showPresets ? ( { @@ -323,43 +187,43 @@ export const DateTimeRangePicker = ({ placeholder="Select Date" /> ) : ( - <> - - - + + + + + setShowPresets(true)} + style={{ alignSelf: 'flex-start' }} + variant="text" + > + Presets + - - + )} ); From 555a0becd84936ff8f540fee7d641219299d2c25 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 16 Dec 2024 14:34:07 -0600 Subject: [PATCH 33/45] Added changeset: Add Date Presets Functionality to Date Picker component --- packages/manager/.changeset/pr-11395-added-1734381247482.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 packages/manager/.changeset/pr-11395-added-1734381247482.md diff --git a/packages/manager/.changeset/pr-11395-added-1734381247482.md b/packages/manager/.changeset/pr-11395-added-1734381247482.md new file mode 100644 index 00000000000..4b1dfb36f7e --- /dev/null +++ b/packages/manager/.changeset/pr-11395-added-1734381247482.md @@ -0,0 +1,5 @@ +--- +"@linode/manager": Added +--- + +Add Date Presets Functionality to Date Picker component ([#11395](https://github.com/linode/manager/pull/11395)) From b5c3606db584bbc04e79db3845ee3675eb2563de Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Mon, 16 Dec 2024 14:50:34 -0600 Subject: [PATCH 34/45] Persist the preset value --- .../src/components/DatePicker/DateTimeRangePicker.tsx | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 0013df716a4..79ff3638f28 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -59,6 +59,10 @@ export const DateTimeRangePicker = ({ const [startDateTime, setStartDateTime] = useState( startDateTimeValue ); + const [presetValue, setPresetValue] = useState({ + label: 'Select a preset', + value: '', + }); const [endDateTime, setEndDateTime] = useState( endDateTimeValue ); @@ -168,6 +172,7 @@ export const DateTimeRangePicker = ({ onChange={(_, selection) => { if (selection) { handlePresetSelection(selection.value); + setPresetValue(selection); } }} options={[ @@ -178,13 +183,10 @@ export const DateTimeRangePicker = ({ { label: 'Last Month', value: 'last_month' }, { label: 'Custom Range', value: 'custom_range' }, ]} - value={{ - label: 'Select a preset', - value: '', - }} fullWidth label="Date Presets" placeholder="Select Date" + value={presetValue} /> ) : ( From bea7bac9e1e94b70dbf222f1a55e9b569a73ddfd Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 17 Dec 2024 15:07:45 -0600 Subject: [PATCH 35/45] Show the start date and end date fields only when custom is selected --- .../components/DatePicker/DateTimePicker.tsx | 21 +++-- .../DatePicker/DateTimeRangePicker.tsx | 77 ++++++++----------- 2 files changed, 50 insertions(+), 48 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index b503ba37674..34cf2d5fbd8 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,6 +1,6 @@ +import { TextField } from '@linode/ui'; import { Divider } from '@linode/ui'; import { Box } from '@linode/ui'; -import { TextField } from '@linode/ui'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; @@ -75,6 +75,8 @@ export const DateTimePicker = ({ value = null, }: DateTimePickerProps) => { const [anchorEl, setAnchorEl] = useState(null); + + // Current and original states const [selectedDateTime, setSelectedDateTime] = useState( value ); @@ -82,6 +84,13 @@ export const DateTimePicker = ({ timeZoneSelectProps.value || null ); + const [originalDateTime, setOriginalDateTime] = useState( + value + ); + const [originalTimeZone, setOriginalTimeZone] = useState( + timeZoneSelectProps.value || null + ); + const TimePickerFieldProps: TextFieldProps = { label: timeSelectProps?.label ?? 'Select Time', noMarginTop: true, @@ -115,6 +124,8 @@ export const DateTimePicker = ({ const handleApply = () => { setAnchorEl(null); + setOriginalDateTime(selectedDateTime); + setOriginalTimeZone(selectedTimeZone); onChange(selectedDateTime); if (onApply) { @@ -124,6 +135,9 @@ export const DateTimePicker = ({ const handleClose = () => { setAnchorEl(null); + setSelectedDateTime(originalDateTime); + setSelectedTimeZone(originalTimeZone); + if (onCancel) { onCancel(); } @@ -247,10 +261,6 @@ export const DateTimePicker = ({ diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 79ff3638f28..2312aeac7f8 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -21,7 +21,8 @@ interface DateTimeRangePickerProps { onChange?: ( start: DateTime | null, end: DateTime | null, - startTimeZone?: null | string + startTimeZone?: null | string, + preset?: string ) => void; /** Whether to show the end timezone field for the end date picker */ showEndTimeZone?: boolean; @@ -35,9 +36,7 @@ interface DateTimeRangePickerProps { startLabel?: string; /** Initial or controlled value for the start timezone */ startTimeZoneValue?: null | string; - /** - * Any additional styles to apply to the root element. - */ + /** Any additional styles to apply to the root element */ sx?: SxProps; } @@ -59,13 +58,10 @@ export const DateTimeRangePicker = ({ const [startDateTime, setStartDateTime] = useState( startDateTimeValue ); - const [presetValue, setPresetValue] = useState({ - label: 'Select a preset', - value: '', - }); const [endDateTime, setEndDateTime] = useState( endDateTimeValue ); + const [presetValue, setPresetValue] = useState(''); const [startTimeZone, setStartTimeZone] = useState( startTimeZoneValue ); @@ -88,37 +84,11 @@ export const DateTimeRangePicker = ({ return; } } - // Reset validation errors if valid + // Reset validation errors setStartDateError(null); setEndDateError(null); }; - const handleStartDateTimeChange = (newStart: DateTime | null) => { - setStartDateTime(newStart); - validateDates(newStart, endDateTime, 'start'); - - if (onChange) { - onChange(newStart, endDateTime, startTimeZone); - } - }; - - const handleEndDateTimeChange = (newEnd: DateTime | null) => { - setEndDateTime(newEnd); - validateDates(startDateTime, newEnd, 'end'); - - if (onChange) { - onChange(startDateTime, newEnd, startTimeZone); - } - }; - - const handleStartTimeZoneChange = (newTimeZone: null | string) => { - setStartTimeZone(newTimeZone); - - if (onChange) { - onChange(startDateTime, endDateTime, newTimeZone); - } - }; - const handlePresetSelection = (value: string) => { const now = DateTime.now(); let newStartDateTime: DateTime | null = null; @@ -156,13 +126,31 @@ export const DateTimeRangePicker = ({ setStartDateTime(newStartDateTime); setEndDateTime(newEndDateTime); + setPresetValue(value); + + if (onChange) { + onChange(newStartDateTime, newEndDateTime, startTimeZone, value); + } + + setShowPresets(value !== 'custom_range'); + }; + + const handleStartDateTimeChange = (newStart: DateTime | null) => { + setStartDateTime(newStart); + validateDates(newStart, endDateTime, 'start'); if (onChange) { - onChange(newStartDateTime, newEndDateTime, startTimeZone); + onChange(newStart, endDateTime, startTimeZone, 'custom_range'); } + }; - // Show date pickers after selecting a preset - setShowPresets(false); + const handleEndDateTimeChange = (newEnd: DateTime | null) => { + setEndDateTime(newEnd); + validateDates(startDateTime, newEnd, 'end'); + + if (onChange) { + onChange(startDateTime, newEnd, startTimeZone, 'custom_range'); + } }; return ( @@ -172,7 +160,6 @@ export const DateTimeRangePicker = ({ onChange={(_, selection) => { if (selection) { handlePresetSelection(selection.value); - setPresetValue(selection); } }} options={[ @@ -181,19 +168,23 @@ export const DateTimeRangePicker = ({ { label: 'Last 30 Days', value: '30days' }, { label: 'This Month', value: 'this_month' }, { label: 'Last Month', value: 'last_month' }, - { label: 'Custom Range', value: 'custom_range' }, + { label: 'Custom', value: 'custom_range' }, ]} + value={ + presetValue + ? { label: presetValue.replace('_', ' '), value: presetValue } + : null + } fullWidth label="Date Presets" - placeholder="Select Date" - value={presetValue} + placeholder="Select a preset" /> ) : ( setStartTimeZone(value), value: startTimeZone, }} errorText={startDateError ?? undefined} From 4ff411b107d55e166c498c4bb746b005e8b35f86 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 17 Dec 2024 23:07:07 -0600 Subject: [PATCH 36/45] Add calendar icon to DateTimePicker component --- .../components/DatePicker/DateTimePicker.tsx | 22 ++++++++++++++++--- 1 file changed, 19 insertions(+), 3 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index 34cf2d5fbd8..f0c085f761f 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,6 +1,7 @@ -import { TextField } from '@linode/ui'; import { Divider } from '@linode/ui'; +import { InputAdornment, TextField } from '@linode/ui'; import { Box } from '@linode/ui'; +import CalendarTodayIcon from '@mui/icons-material/CalendarToday'; import { Grid, Popover } from '@mui/material'; import { AdapterLuxon } from '@mui/x-date-pickers/AdapterLuxon'; import { DateCalendar } from '@mui/x-date-pickers/DateCalendar'; @@ -66,7 +67,7 @@ export const DateTimePicker = ({ onApply, onCancel, onChange, - placeholder = 'yyyy-MM-dd HH:mm', + placeholder = 'Select Date', showTime = true, showTimeZone = true, sx, @@ -153,6 +154,22 @@ export const DateTimePicker = ({ + + + ), + sx: { paddingLeft: '32px' }, + }} value={ selectedDateTime ? `${selectedDateTime.toFormat(format)}${ @@ -160,7 +177,6 @@ export const DateTimePicker = ({ }` : '' } - InputProps={{ readOnly: true }} errorText={errorText} label={label} onClick={(event) => setAnchorEl(event.currentTarget)} From 510485a7ff1415bfe1736092ea3c46e75d908987 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 18 Dec 2024 01:20:04 -0600 Subject: [PATCH 37/45] code cleanup and adjust tests --- .../DatePicker/DateTimeRangePicker.test.tsx | 286 ++++++++---------- .../DatePicker/DateTimeRangePicker.tsx | 33 +- 2 files changed, 144 insertions(+), 175 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index 83f9191d92c..76c208b344d 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -10,7 +10,19 @@ import { DateTimeRangePicker } from './DateTimeRangePicker'; describe('DateTimeRangePicker Component', () => { const onChangeMock = vi.fn(); + let fixedNow: DateTime; + beforeEach(() => { + // Mock DateTime.now to return a fixed datetime + fixedNow = DateTime.fromISO( + '2024-12-18T00:28:27.071-06:00' + ) as DateTime; + vi.spyOn(DateTime, 'now').mockImplementation(() => fixedNow); + }); + + afterEach(() => { + // Restore the original DateTime.now implementation after each test + vi.restoreAllMocks(); vi.clearAllMocks(); }); @@ -30,12 +42,13 @@ describe('DateTimeRangePicker Component', () => { await userEvent.click(screen.getByRole('gridcell', { name: '10' })); await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - // Check if the onChange function is called with the expected DateTime value - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ day: 10 }), - null, - null - ); + // Check if the onChange function is called with the expected value + expect(onChangeMock).toHaveBeenCalledWith({ + end: null, + preset: 'custom_range', + start: '2024-12-10T00:00:00.000-06:00', + timeZone: null, + }); }); it('should show error when end date-time is before start date-time', async () => { @@ -51,11 +64,11 @@ describe('DateTimeRangePicker Component', () => { const endDateField = screen.getByLabelText('End Date and Time'); await userEvent.click(endDateField); - // Check if the date before the start date is disabled via a class or attribute + // Set start date-time to the 10th await userEvent.click(screen.getByRole('gridcell', { name: '10' })); await userEvent.click(screen.getByRole('button', { name: 'Apply' })); - // Confirm error message is not shown since the click was blocked + // Confirm error message is displayed expect( screen.getByText('End date/time cannot be before the start date/time.') ).toBeInTheDocument(); @@ -128,40 +141,20 @@ describe('DateTimeRangePicker Component', () => { const last24HoursOption = screen.getByText('Last 24 Hours'); await userEvent.click(last24HoursOption); - // Verify that onChange is called with the correct date range - const now = DateTime.now(); - const expectedStartDateTime = now.minus({ hours: 24 }); - - const expectedEndDateTime = now; - - const expectedStartDateValue = expectedStartDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - const expectedEndDateValue = expectedEndDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ - day: expectedStartDateTime.day, - month: expectedStartDateTime.month, - year: expectedStartDateTime.year, - }), - expect.objectContaining({ - day: expectedEndDateTime.day, - month: expectedEndDateTime.month, - year: expectedEndDateTime.year, - }), - null - ); - - // Verify that Date input fields has the correct date range + // Expected start and end dates in ISO format + const expectedStartDateISO = fixedNow.minus({ hours: 24 }).toISO(); // 2024-12-17T00:28:27.071-06:00 + const expectedEndDateISO = fixedNow.toISO(); // 2024-12-18T00:28:27.071-06:00 + + // Verify onChangeMock was called with correct ISO strings + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: '24hours', + start: expectedStartDateISO, + timeZone: null, + }); expect( - screen.getByRole('textbox', { name: 'Start Date and Time' }) - ).toHaveValue(expectedStartDateValue); - expect( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ).toHaveValue(expectedEndDateValue); + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); }); it('should set the date range for the last 7 days when the "Last 7 Days" preset is selected', async () => { @@ -177,39 +170,20 @@ describe('DateTimeRangePicker Component', () => { const last7DaysOption = screen.getByText('Last 7 Days'); await userEvent.click(last7DaysOption); - // Verify that onChange is called with the correct date range - const now = DateTime.now(); - const expectedStartDateTime = now.minus({ days: 7 }); - const expectedEndDateTime = now; + // Expected start and end dates in ISO format + const expectedStartDateISO = fixedNow.minus({ days: 7 }).toISO(); + const expectedEndDateISO = fixedNow.toISO(); - const expectedStartDateValue = expectedStartDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - const expectedEndDateValue = expectedEndDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ - day: expectedStartDateTime.day, - month: expectedStartDateTime.month, - year: expectedStartDateTime.year, - }), - expect.objectContaining({ - day: expectedEndDateTime.day, - month: expectedEndDateTime.month, - year: expectedEndDateTime.year, - }), - null - ); - - // Verify that Date input fields have the correct date range - expect( - screen.getByRole('textbox', { name: 'Start Date and Time' }) - ).toHaveValue(expectedStartDateValue); + // Verify that onChange is called with the correct date range + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: '7days', + start: expectedStartDateISO, + timeZone: null, + }); expect( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ).toHaveValue(expectedEndDateValue); + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); }); it('should set the date range for the last 30 days when the "Last 30 Days" preset is selected', async () => { @@ -225,40 +199,20 @@ describe('DateTimeRangePicker Component', () => { const last30DaysOption = screen.getByText('Last 30 Days'); await userEvent.click(last30DaysOption); - const now = DateTime.now(); - const expectedStartDateTime = now.minus({ days: 30 }); - const expectedEndDateTime = now; - - // Use the same format as the component for verification - const expectedStartDateValue = expectedStartDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - const expectedEndDateValue = expectedEndDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); + // Expected start and end dates in ISO format + const expectedStartDateISO = fixedNow.minus({ days: 30 }).toISO(); + const expectedEndDateISO = fixedNow.toISO(); // Verify that onChange is called with the correct date range - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ - day: expectedStartDateTime.day, - month: expectedStartDateTime.month, - year: expectedStartDateTime.year, - }), - expect.objectContaining({ - day: expectedEndDateTime.day, - month: expectedEndDateTime.month, - year: expectedEndDateTime.year, - }), - null - ); - - // Verify the input fields display the correct values - expect( - screen.getByRole('textbox', { name: 'Start Date and Time' }) - ).toHaveValue(expectedStartDateValue); + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: '30days', + start: expectedStartDateISO, + timeZone: null, + }); expect( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ).toHaveValue(expectedEndDateValue); + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); }); it('should set the date range for this month when the "This Month" preset is selected', async () => { @@ -274,40 +228,20 @@ describe('DateTimeRangePicker Component', () => { const thisMonthOption = screen.getByText('This Month'); await userEvent.click(thisMonthOption); - const now = DateTime.now(); - const expectedStartDateTime = now.startOf('month'); - const expectedEndDateTime = now.endOf('month'); - - // Use the same format as the component for verification - const expectedStartDateValue = expectedStartDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - const expectedEndDateValue = expectedEndDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); + // Expected start and end dates in ISO format + const expectedStartDateISO = fixedNow.startOf('month').toISO(); + const expectedEndDateISO = fixedNow.endOf('month').toISO(); // Verify that onChange is called with the correct date range - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ - day: expectedStartDateTime.day, - month: expectedStartDateTime.month, - year: expectedStartDateTime.year, - }), - expect.objectContaining({ - day: expectedEndDateTime.day, - month: expectedEndDateTime.month, - year: expectedEndDateTime.year, - }), - null - ); - - // Verify the input fields display the correct values - expect( - screen.getByRole('textbox', { name: 'Start Date and Time' }) - ).toHaveValue(expectedStartDateValue); + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: 'this_month', + start: expectedStartDateISO, + timeZone: null, + }); expect( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ).toHaveValue(expectedEndDateValue); + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); }); it('should set the date range for last month when the "Last Month" preset is selected', async () => { @@ -323,41 +257,22 @@ describe('DateTimeRangePicker Component', () => { const lastMonthOption = screen.getByText('Last Month'); await userEvent.click(lastMonthOption); - const now = DateTime.now(); - const lastMonth = now.minus({ months: 1 }); - const expectedStartDateTime = lastMonth.startOf('month'); - const expectedEndDateTime = lastMonth.endOf('month'); + const lastMonth = fixedNow.minus({ months: 1 }); - // Use the same format as the component for verification - const expectedStartDateValue = expectedStartDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); - const expectedEndDateValue = expectedEndDateTime.toFormat( - 'yyyy-MM-dd HH:mm' - ); + // Expected start and end dates in ISO format + const expectedStartDateISO = lastMonth.startOf('month').toISO(); + const expectedEndDateISO = lastMonth.endOf('month').toISO(); // Verify that onChange is called with the correct date range - expect(onChangeMock).toHaveBeenCalledWith( - expect.objectContaining({ - day: expectedStartDateTime.day, - month: expectedStartDateTime.month, - year: expectedStartDateTime.year, - }), - expect.objectContaining({ - day: expectedEndDateTime.day, - month: expectedEndDateTime.month, - year: expectedEndDateTime.year, - }), - null - ); - - // Verify the input fields display the correct values - expect( - screen.getByRole('textbox', { name: 'Start Date and Time' }) - ).toHaveValue(expectedStartDateValue); + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: 'last_month', + start: expectedStartDateISO, + timeZone: null, + }); expect( - screen.getByRole('textbox', { name: 'End Date and Time' }) - ).toHaveValue(expectedEndDateValue); + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); }); it('should display the date range fields with empty values when the "Custom Range" preset is selected', async () => { @@ -370,7 +285,7 @@ describe('DateTimeRangePicker Component', () => { await userEvent.click(presetsDropdown); // Select the "Custom Range" option - const customRange = screen.getByText('Custom Range'); + const customRange = screen.getByText('Custom'); await userEvent.click(customRange); // Verify the input fields display the correct values @@ -381,5 +296,44 @@ describe('DateTimeRangePicker Component', () => { screen.getByRole('textbox', { name: 'End Date and Time' }) ).toHaveValue(''); expect(screen.getByRole('button', { name: 'Presets' })).toBeInTheDocument(); + + // Set start date-time to the 15th + const startDateField = screen.getByLabelText('Start Date and Time'); + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '15' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Open the end date picker + const endDateField = screen.getByLabelText('End Date and Time'); + await userEvent.click(endDateField); + + // Set start date-time to the 12th + await userEvent.click(screen.getByRole('gridcell', { name: '12' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Confirm error message is shown since the click was blocked + expect( + screen.getByText('End date/time cannot be before the start date/time.') + ).toBeInTheDocument(); + + // Set start date-time to the 11th + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '11' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Confirm error message is not displayed + expect( + screen.queryByText('End date/time cannot be before the start date/time.') + ).not.toBeInTheDocument(); + + // Set start date-time to the 20th + await userEvent.click(startDateField); + await userEvent.click(screen.getByRole('gridcell', { name: '20' })); + await userEvent.click(screen.getByRole('button', { name: 'Apply' })); + + // Confirm error message is not displayed + expect( + screen.queryByText('Start date/time cannot be after the end date/time.') + ).toBeInTheDocument(); }); }); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 2312aeac7f8..aae29d698fb 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -18,12 +18,12 @@ interface DateTimeRangePickerProps { /** Format for displaying the date-time */ format?: string; /** Callback when the date-time range changes */ - onChange?: ( - start: DateTime | null, - end: DateTime | null, - startTimeZone?: null | string, - preset?: string - ) => void; + onChange?: (params: { + end: null | string; + preset?: string; + start: null | string; + timeZone?: null | string; + }) => void; /** Whether to show the end timezone field for the end date picker */ showEndTimeZone?: boolean; /** Whether to show the start timezone field for the start date picker */ @@ -129,7 +129,12 @@ export const DateTimeRangePicker = ({ setPresetValue(value); if (onChange) { - onChange(newStartDateTime, newEndDateTime, startTimeZone, value); + onChange({ + end: newEndDateTime?.toISO() ?? null, + preset: value, + start: newStartDateTime?.toISO() ?? null, + timeZone: startTimeZone, + }); } setShowPresets(value !== 'custom_range'); @@ -140,7 +145,12 @@ export const DateTimeRangePicker = ({ validateDates(newStart, endDateTime, 'start'); if (onChange) { - onChange(newStart, endDateTime, startTimeZone, 'custom_range'); + onChange({ + end: endDateTime?.toISO() ?? null, + preset: 'custom_range', + start: newStart?.toISO() ?? null, + timeZone: startTimeZone, + }); } }; @@ -149,7 +159,12 @@ export const DateTimeRangePicker = ({ validateDates(startDateTime, newEnd, 'end'); if (onChange) { - onChange(startDateTime, newEnd, startTimeZone, 'custom_range'); + onChange({ + end: newEnd?.toISO() ?? null, + preset: 'custom_range', + start: startDateTime?.toISO() ?? null, + timeZone: startTimeZone, + }); } }; From 0cf080f28f3045181337a05e3b5c4da31144a6cd Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 18 Dec 2024 07:01:01 -0600 Subject: [PATCH 38/45] Update packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx Co-authored-by: Nikhil Agrawal <165884194+nikhagra-akamai@users.noreply.github.com> --- .../manager/src/components/DatePicker/DateTimeRangePicker.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index aae29d698fb..287c587f7c7 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -190,6 +190,7 @@ export const DateTimeRangePicker = ({ ? { label: presetValue.replace('_', ' '), value: presetValue } : null } + isOptionEqualToValue={(option, value) => option.value === value.value} fullWidth label="Date Presets" placeholder="Select a preset" From 729c18472e567d284f6c65e0de662cd97eae9804 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 18 Dec 2024 07:45:23 -0600 Subject: [PATCH 39/45] update components --- packages/manager/src/components/DatePicker/DateTimePicker.tsx | 1 + .../manager/src/components/DatePicker/DateTimeRangePicker.tsx | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index f0c085f761f..86c66ee834a 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -179,6 +179,7 @@ export const DateTimePicker = ({ } errorText={errorText} label={label} + noMarginTop onClick={(event) => setAnchorEl(event.currentTarget)} placeholder={placeholder} /> diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 287c587f7c7..5d92c539158 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -190,9 +190,10 @@ export const DateTimeRangePicker = ({ ? { label: presetValue.replace('_', ' '), value: presetValue } : null } - isOptionEqualToValue={(option, value) => option.value === value.value} fullWidth + isOptionEqualToValue={(option, value) => option.value === value.value} label="Date Presets" + noMarginTop placeholder="Select a preset" /> ) : ( From 727c59c516acb43eb667ba9598d8bbce24a1b168 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 18 Dec 2024 11:15:36 -0600 Subject: [PATCH 40/45] Organize and additional prop support to DateTimeRangePicker component --- .../DateTimeRangePicker.stories.tsx | 194 ++++++++++++------ .../DatePicker/DateTimeRangePicker.test.tsx | 81 ++++---- .../DatePicker/DateTimeRangePicker.tsx | 129 ++++++++---- 3 files changed, 269 insertions(+), 135 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx index f34ac6b190b..aeaecd516d0 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.stories.tsx @@ -10,61 +10,111 @@ type Story = StoryObj; export const Default: Story = { args: { - endDateErrorMessage: '', - endDateTimeValue: null, - endLabel: 'End Date and Time', + endDateProps: { + errorMessage: '', + label: 'End Date and Time', + placeholder: '', + showTimeZone: false, + value: null, + }, format: 'yyyy-MM-dd HH:mm', onChange: action('DateTime range changed'), - showEndTimeZone: true, - showStartTimeZone: true, - startDateErrorMessage: '', - startDateTimeValue: null, - startLabel: 'Start Date and Time', - startTimeZoneValue: null, + presetsProps: { + defaultValue: { label: '', value: '' }, + enablePresets: true, + label: '', + placeholder: '', + }, + startDateProps: { + errorMessage: '', + label: 'Start Date and Time', + placeholder: '', + showTimeZone: true, + timeZoneValue: null, + value: null, + }, + sx: {}, }, render: (args) => , }; export const WithInitialValues: Story = { args: { - endDateTimeValue: DateTime.now(), - endLabel: 'End Date and Time', + endDateProps: { + label: 'End Date and Time', + showTimeZone: true, + value: DateTime.now(), + }, format: 'yyyy-MM-dd HH:mm', onChange: action('DateTime range changed'), - showEndTimeZone: true, - showStartTimeZone: true, - startDateTimeValue: DateTime.now().minus({ days: 1 }), - startLabel: 'Start Date and Time', - startTimeZoneValue: 'America/New_York', + presetsProps: { + defaultValue: { label: 'Last 7 Days', value: '7days' }, + enablePresets: true, + label: 'Time Range', + placeholder: 'Select Range', + }, + startDateProps: { + label: 'Start Date and Time', + showTimeZone: true, + timeZoneValue: 'America/New_York', + value: DateTime.now().minus({ days: 1 }), + }, + sx: {}, }, }; export const WithCustomErrors: Story = { args: { - endDateErrorMessage: 'End date must be after the start date.', - endDateTimeValue: DateTime.now().minus({ days: 1 }), - endLabel: 'Custom End Label', + endDateProps: { + errorMessage: 'End date must be after the start date.', + label: 'Custom End Label', + placeholder: '', + showTimeZone: false, + value: DateTime.now().minus({ days: 1 }), + }, format: 'yyyy-MM-dd HH:mm', onChange: action('DateTime range changed'), - startDateErrorMessage: 'Start date must be before the end date.', - startDateTimeValue: DateTime.now().minus({ days: 2 }), - startLabel: 'Custom Start Label', + presetsProps: { + defaultValue: { label: '', value: '' }, + enablePresets: true, + label: '', + placeholder: '', + }, + startDateProps: { + errorMessage: 'Start date must be before the end date.', + label: 'Start Date and Time', + placeholder: '', + showTimeZone: true, + timeZoneValue: null, + value: DateTime.now().minus({ days: 2 }), + }, }, }; const meta: Meta = { argTypes: { - endDateErrorMessage: { - control: 'text', - description: 'Custom error message for invalid end date', - }, - endDateTimeValue: { - control: 'date', - description: 'Initial or controlled value for the end date-time', - }, - endLabel: { - control: 'text', - description: 'Custom label for the end date-time picker', + endDateProps: { + errorMessage: { + control: 'text', + description: 'Custom error message for invalid end date', + }, + label: { + control: 'text', + description: 'Custom label for the end date-time picker', + }, + placeholder: { + control: 'text', + description: 'Placeholder for the end date-time', + }, + showTimeZone: { + control: 'boolean', + description: + 'Whether to show the timezone selector for the end date picker', + }, + value: { + control: 'date', + description: 'Initial or controlled value for the end date-time', + }, }, format: { control: 'text', @@ -74,31 +124,57 @@ const meta: Meta = { action: 'DateTime range changed', description: 'Callback when the date-time range changes', }, - showEndTimeZone: { - control: 'boolean', - description: - 'Whether to show the timezone selector for the end date picker', - }, - showStartTimeZone: { - control: 'boolean', - description: - 'Whether to show the timezone selector for the start date picker', - }, - startDateErrorMessage: { - control: 'text', - description: 'Custom error message for invalid start date', - }, - startDateTimeValue: { - control: 'date', - description: 'Initial or controlled value for the start date-time', + presetsProps: { + defaultValue: { + label: { + control: 'text', + description: 'Default value label for the presets field', + }, + value: { + control: 'text', + description: 'Default value for the presets field', + }, + }, + enablePresets: { + control: 'boolean', + description: + 'If true, shows the date presets field instead of the date pickers', + }, + label: { + control: 'text', + description: 'Label for the presets dropdown', + }, + placeholder: { + control: 'text', + description: 'Placeholder for the presets dropdown', + }, }, - startLabel: { - control: 'text', - description: 'Custom label for the start date-time picker', - }, - startTimeZoneValue: { - control: 'text', - description: 'Initial or controlled value for the start timezone', + startDateProps: { + errorMessage: { + control: 'text', + description: 'Custom error message for invalid start date', + }, + placeholder: { + control: 'text', + description: 'Placeholder for the start date-time', + }, + showTimeZone: { + control: 'boolean', + description: + 'Whether to show the timezone selector for the start date picker', + }, + startLabel: { + control: 'text', + description: 'Custom label for the start date-time picker', + }, + timeZoneValue: { + control: 'text', + description: 'Initial or controlled value for the start timezone', + }, + value: { + control: 'date', + description: 'Initial or controlled value for the start date-time', + }, }, sx: { control: 'object', @@ -106,9 +182,9 @@ const meta: Meta = { }, }, args: { - endLabel: 'End Date and Time', + endDateProps: { label: 'End Date and Time' }, format: 'yyyy-MM-dd HH:mm', - startLabel: 'Start Date and Time', + startDateProps: { label: 'Start Date and Time' }, }, component: DateTimeRangePicker, title: 'Components/DatePicker/DateTimeRangePicker', diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index 76c208b344d..42dc4c05830 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -7,9 +7,26 @@ import { renderWithTheme } from 'src/utilities/testHelpers'; import { DateTimeRangePicker } from './DateTimeRangePicker'; -describe('DateTimeRangePicker Component', () => { - const onChangeMock = vi.fn(); +import type { DateTimeRangePickerProps } from './DateTimeRangePicker'; + +const onChangeMock = vi.fn(); + +const Props: DateTimeRangePickerProps = { + endDateProps: { + label: 'End Date and Time', + }, + onChange: onChangeMock, + presetsProps: { + enablePresets: true, + label: 'Date Presets', + }, + startDateProps: { + label: 'Start Date and Time', + }, +}; + +describe('DateTimeRangePicker Component', () => { let fixedNow: DateTime; beforeEach(() => { @@ -75,13 +92,11 @@ describe('DateTimeRangePicker Component', () => { }); it('should show error when start date-time is after end date-time', async () => { - renderWithTheme( - - ); + const updateProps = { + ...Props, + presetsProps: { ...Props.presetsProps, enablePresets: false }, + }; + renderWithTheme(); // Set the end date-time to the 15th const endDateField = screen.getByLabelText('End Date and Time'); @@ -102,15 +117,21 @@ describe('DateTimeRangePicker Component', () => { }); it('should display custom error messages when start date-time is after end date-time', async () => { - renderWithTheme( - - ); + const updatedProps = { + ...Props, + endDateProps: { + ...Props.endDateProps, + errorMessage: 'Custom end date error', + label: 'End Date and Time', + }, + presetsProps: {}, + startDateProps: { + ...Props.startDateProps, + errorMessage: 'Custom start date error', + label: 'Start Date and Time', + }, + }; + renderWithTheme(); // Set the end date-time to the 15th const endDateField = screen.getByLabelText('End Date and Time'); @@ -129,9 +150,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should set the date range for the last 24 hours when the "Last 24 Hours" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); @@ -158,9 +177,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should set the date range for the last 7 days when the "Last 7 Days" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); @@ -187,9 +204,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should set the date range for the last 30 days when the "Last 30 Days" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); @@ -216,9 +231,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should set the date range for this month when the "This Month" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); @@ -245,9 +258,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should set the date range for last month when the "Last Month" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); @@ -276,9 +287,7 @@ describe('DateTimeRangePicker Component', () => { }); it('should display the date range fields with empty values when the "Custom Range" preset is selected', async () => { - renderWithTheme( - - ); + renderWithTheme(); // Open the presets dropdown const presetsDropdown = screen.getByLabelText('Date Presets'); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 5d92c539158..72e4d759043 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -6,55 +6,98 @@ import { DateTimePicker } from './DateTimePicker'; import type { SxProps, Theme } from '@mui/material/styles'; -interface DateTimeRangePickerProps { - /** If true, shows the date presets field instead of the date pickers */ - enablePresets?: boolean; - /** Custom error message for invalid end date */ - endDateErrorMessage?: string; - /** Initial or controlled value for the end date-time */ - endDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - endLabel?: string; +export interface DateTimeRangePickerProps { + /** Properties for the end date field */ + endDateProps?: { + /** Custom error message for invalid end date */ + errorMessage?: string; + /** Label for the end date field */ + label?: string; + /** placeholder for the end date field */ + placeholder?: string; + /** Whether to show the timezone selector for the end date */ + showTimeZone?: boolean; + /** Initial or controlled value for the end date-time */ + value?: DateTime | null; + }; + /** Format for displaying the date-time */ format?: string; - /** Callback when the date-time range changes */ + + /** Callback when the date-time range changes, + * this returns start date, end date in ISO formate, + * preset value and timezone + * */ onChange?: (params: { end: null | string; preset?: string; start: null | string; timeZone?: null | string; }) => void; - /** Whether to show the end timezone field for the end date picker */ - showEndTimeZone?: boolean; - /** Whether to show the start timezone field for the start date picker */ - showStartTimeZone?: boolean; - /** Custom error message for invalid start date */ - startDateErrorMessage?: string; - /** Initial or controlled value for the start date-time */ - startDateTimeValue?: DateTime | null; - /** Custom labels for the start and end date/time fields */ - startLabel?: string; - /** Initial or controlled value for the start timezone */ - startTimeZoneValue?: null | string; + + /** Additional settings for the presets dropdown */ + presetsProps?: { + /** Default value for the presets field */ + defaultValue?: { label: string; value: string }; + /** If true, shows the date presets field instead of the date pickers */ + enablePresets?: boolean; + /** Label for the presets field */ + label?: string; + /** placeholder for the presets field */ + placeholder?: string; + }; + + /** Properties for the start date field */ + startDateProps?: { + /** Custom error message for invalid start date */ + errorMessage?: string; + /** Label for the start date field */ + label?: string; + /** placeholder for the start date field */ + placeholder?: string; + /** Whether to show the timezone selector for the start date */ + showTimeZone?: boolean; + /** Initial or controlled value for the start timezone */ + timeZoneValue?: null | string; + /** Initial or controlled value for the start date-time */ + value?: DateTime | null; + }; + /** Any additional styles to apply to the root element */ sx?: SxProps; } -export const DateTimeRangePicker = ({ - enablePresets = false, - endDateErrorMessage = 'End date/time cannot be before the start date/time.', - endDateTimeValue = null, - endLabel = 'End Date and Time', - format = 'yyyy-MM-dd HH:mm', - onChange, - showEndTimeZone = false, - showStartTimeZone = false, - startDateErrorMessage = 'Start date/time cannot be after the end date/time.', - startDateTimeValue = null, - startLabel = 'Start Date and Time', - startTimeZoneValue = null, - sx, -}: DateTimeRangePickerProps) => { +export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { + const { + endDateProps: { + errorMessage: endDateErrorMessage = 'End date/time cannot be before the start date/time.', + label: endLabel = 'End Date and Time', + placeholder: endDatePlaceholder, + showTimeZone: showEndTimeZone = false, + value: endDateTimeValue = null, + } = {}, + + format = 'yyyy-MM-dd HH:mm', + + onChange, + + presetsProps: { + defaultValue: presetsDefaultValue, + enablePresets = false, + label: presetsLabel = 'Time Range', + placeholder: presetsPlaceholder = 'Select a preset', + } = {}, + startDateProps: { + errorMessage: startDateErrorMessage = 'Start date/time cannot be after the end date/time.', + label: startLabel = 'Start Date and Time', + placeholder: startDatePlaceholder, + showTimeZone: showStartTimeZone = false, + timeZoneValue: startTimeZoneValue = null, + value: startDateTimeValue = null, + } = {}, + sx, + } = props; + const [startDateTime, setStartDateTime] = useState( startDateTimeValue ); @@ -190,11 +233,12 @@ export const DateTimeRangePicker = ({ ? { label: presetValue.replace('_', ' '), value: presetValue } : null } + defaultValue={presetsDefaultValue} fullWidth isOptionEqualToValue={(option, value) => option.value === value.value} - label="Date Presets" + label={presetsLabel} noMarginTop - placeholder="Select a preset" + placeholder={presetsPlaceholder} /> ) : ( @@ -208,6 +252,7 @@ export const DateTimeRangePicker = ({ format={format} label={startLabel} onChange={handleStartDateTimeChange} + placeholder={startDatePlaceholder} showTimeZone={showStartTimeZone} timeSelectProps={{ label: 'Start Time' }} value={startDateTime} @@ -220,13 +265,17 @@ export const DateTimeRangePicker = ({ format={format} label={endLabel} onChange={handleEndDateTimeChange} + placeholder={endDatePlaceholder} showTimeZone={showEndTimeZone} timeSelectProps={{ label: 'End Time' }} value={endDateTime} /> setShowPresets(true)} + onClick={() => { + setShowPresets(true); + setPresetValue(''); + }} style={{ alignSelf: 'flex-start' }} variant="text" > From 9ebfb7f790ce5becf0c97d89b0177e5f9a27d82e Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 18 Dec 2024 12:03:56 -0600 Subject: [PATCH 41/45] Code cleanup --- .../DatePicker/DateTimeRangePicker.tsx | 50 ++++++++++++------- 1 file changed, 32 insertions(+), 18 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 72e4d759043..2ae5a077691 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,4 +1,6 @@ import { Autocomplete, Box, StyledActionButton } from '@linode/ui'; +import { useTheme } from '@mui/material'; +import useMediaQuery from '@mui/material/useMediaQuery'; import { DateTime } from 'luxon'; import React, { useState } from 'react'; @@ -67,6 +69,15 @@ export interface DateTimeRangePickerProps { sx?: SxProps; } +const presetsOptions = [ + { label: 'Last 24 Hours', value: '24hours' }, + { label: 'Last 7 Days', value: '7days' }, + { label: 'Last 30 Days', value: '30days' }, + { label: 'This Month', value: 'this_month' }, + { label: 'Last Month', value: 'last_month' }, + { label: 'Custom', value: 'custom_range' }, +]; + export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { const { endDateProps: { @@ -82,7 +93,7 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { onChange, presetsProps: { - defaultValue: presetsDefaultValue, + defaultValue: presetsDefaultValue = { label: '', value: '' }, enablePresets = false, label: presetsLabel = 'Time Range', placeholder: presetsPlaceholder = 'Select a preset', @@ -104,7 +115,10 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { const [endDateTime, setEndDateTime] = useState( endDateTimeValue ); - const [presetValue, setPresetValue] = useState(''); + const [presetValue, setPresetValue] = useState<{ + label: string; + value: string; + }>(presetsDefaultValue); const [startTimeZone, setStartTimeZone] = useState( startTimeZoneValue ); @@ -112,6 +126,9 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { const [endDateError, setEndDateError] = useState(null); const [showPresets, setShowPresets] = useState(enablePresets); + const theme = useTheme(); + const isSmallScreen = useMediaQuery(theme.breakpoints.down('md')); + const validateDates = ( start: DateTime | null, end: DateTime | null, @@ -169,7 +186,10 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { setStartDateTime(newStartDateTime); setEndDateTime(newEndDateTime); - setPresetValue(value); + setPresetValue( + presetsOptions.find((option) => option.value === value) ?? + presetsDefaultValue + ); if (onChange) { onChange({ @@ -220,28 +240,22 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { handlePresetSelection(selection.value); } }} - options={[ - { label: 'Last 24 Hours', value: '24hours' }, - { label: 'Last 7 Days', value: '7days' }, - { label: 'Last 30 Days', value: '30days' }, - { label: 'This Month', value: 'this_month' }, - { label: 'Last Month', value: 'last_month' }, - { label: 'Custom', value: 'custom_range' }, - ]} - value={ - presetValue - ? { label: presetValue.replace('_', ' '), value: presetValue } - : null - } defaultValue={presetsDefaultValue} + disableClearable fullWidth isOptionEqualToValue={(option, value) => option.value === value.value} label={presetsLabel} noMarginTop + options={presetsOptions} placeholder={presetsPlaceholder} + value={presetValue} /> ) : ( - + { { setShowPresets(true); - setPresetValue(''); + setPresetValue(presetsDefaultValue); }} style={{ alignSelf: 'flex-start' }} variant="text" From e8562757158fa01428a7b4b89d22f77dd5f6f692 Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Tue, 7 Jan 2025 11:58:22 -0600 Subject: [PATCH 42/45] PR feedback - @coliu-akamai --- .../DatePicker/DateTimeRangePicker.test.tsx | 5 ++++- .../DatePicker/DateTimeRangePicker.tsx | 22 ++++++++++++++----- .../Linodes/LinodeEntityDetailHeader.tsx | 9 ++++++++ 3 files changed, 29 insertions(+), 7 deletions(-) diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx index 42dc4c05830..02bdd4f808b 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -51,6 +51,9 @@ describe('DateTimeRangePicker Component', () => { }); it('should call onChange when start date is changed', async () => { + const currentYear = new Date().getFullYear(); // Dynamically get the current year + const currentMonth = String(new Date().getMonth() + 1).padStart(2, '0'); // Get current month (1-based) + renderWithTheme(); // Open start date picker @@ -63,7 +66,7 @@ describe('DateTimeRangePicker Component', () => { expect(onChangeMock).toHaveBeenCalledWith({ end: null, preset: 'custom_range', - start: '2024-12-10T00:00:00.000-06:00', + start: `${currentYear}-${currentMonth}-10T00:00:00.000-06:00`, timeZone: null, }); }); diff --git a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx index 2ae5a077691..7083ba7bee8 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -69,7 +69,15 @@ export interface DateTimeRangePickerProps { sx?: SxProps; } -const presetsOptions = [ +type DatePresetType = + | '7days' + | '24hours' + | '30days' + | 'custom_range' + | 'last_month' + | 'this_month'; + +const presetsOptions: { label: string; value: DatePresetType }[] = [ { label: 'Last 24 Hours', value: '24hours' }, { label: 'Last 7 Days', value: '7days' }, { label: 'Last 30 Days', value: '30days' }, @@ -149,7 +157,7 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { setEndDateError(null); }; - const handlePresetSelection = (value: string) => { + const handlePresetSelection = (value: DatePresetType) => { const now = DateTime.now(); let newStartDateTime: DateTime | null = null; let newEndDateTime: DateTime | null = null; @@ -237,13 +245,12 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { { if (selection) { - handlePresetSelection(selection.value); + handlePresetSelection(selection.value as DatePresetType); } }} defaultValue={presetsDefaultValue} disableClearable fullWidth - isOptionEqualToValue={(option, value) => option.value === value.value} label={presetsLabel} noMarginTop options={presetsOptions} @@ -284,13 +291,16 @@ export const DateTimeRangePicker = (props: DateTimeRangePickerProps) => { timeSelectProps={{ label: 'End Time' }} value={endDateTime} /> - + { setShowPresets(true); setPresetValue(presetsDefaultValue); }} - style={{ alignSelf: 'flex-start' }} variant="text" > Presets diff --git a/packages/manager/src/features/Linodes/LinodeEntityDetailHeader.tsx b/packages/manager/src/features/Linodes/LinodeEntityDetailHeader.tsx index d21c7b67087..37f44a86813 100644 --- a/packages/manager/src/features/Linodes/LinodeEntityDetailHeader.tsx +++ b/packages/manager/src/features/Linodes/LinodeEntityDetailHeader.tsx @@ -3,6 +3,7 @@ import { Typography } from '@mui/material'; import { useTheme } from '@mui/material/styles'; import * as React from 'react'; +import { DateTimeRangePicker } from 'src/components/DatePicker/DateTimeRangePicker'; import { EntityHeader } from 'src/components/EntityHeader/EntityHeader'; import { Hidden } from 'src/components/Hidden'; import { StatusIcon } from 'src/components/StatusIcon/StatusIcon'; @@ -147,6 +148,13 @@ export const LinodeEntityDetailHeader = ( title={{linodeLabel}} variant={variant} > + {}} + presetsProps={{ enablePresets: true }} + startDateProps={{ label: 'Start Date and Time', value: null }} + /> )} +