From e97ff2f18e74e73d058f2365b486d334c494ee7f Mon Sep 17 00:00:00 2001 From: cpathipa <119517080+cpathipa@users.noreply.github.com> Date: Wed, 8 Jan 2025 15:14:03 -0600 Subject: [PATCH] feat: [M3-8936] - Add Date Presets Functionality to Date Picker component. (#11395) * unit test coverage for HostNameTableCell * Revert "unit test coverage for HostNameTableCell" This reverts commit b274baf67e27d79fd4e764607ded7c5aa755ee8b. * chore: [M3-8662] - Update Github Actions actions (#11009) * update actions * add changeset --------- Co-authored-by: Banks Nussman * Basci date picker component * Test coverage for date picker component * DatePicker Stories * Custom DateTimePicker component * Reusable TimeZone Select Component * Create custom DateTimeRangePicker component * Storybook for DateTimePicker * Fix tests and remove console warnings * changeset * Update packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx Co-authored-by: Connie Liu <139280159+coliu-akamai@users.noreply.github.com> * Adjust styles for DatePicker * Adjust styles for DateTimePicker * update imports * Render time and timezone conditionally in DateTimePicker component * Move DatePicker to UI package * Add DatePicker dependencies * Code cleanup * PR feedback * code cleanup * Move DatePicker back to src/components * Reverting changes * Code cleanup * Adjust broken tests * Update TimeZoneSelect.tsx * Code cleanup * Add validation for start date agains end date. * Adjust styles for TimePicker component. * Add the functionality to support Date Presets * Update presets functionality and add test coverage. * Added changeset: Add Date Presets Functionality to Date Picker component * Persist the preset value * Show the start date and end date fields only when custom is selected * Add calendar icon to DateTimePicker component * code cleanup and adjust tests * Update packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx Co-authored-by: Nikhil Agrawal <165884194+nikhagra-akamai@users.noreply.github.com> * update components * Organize and additional prop support to DateTimeRangePicker component * Code cleanup * PR feedback - @coliu-akamai * Move styles to theme level * Revert "Move styles to theme level" This reverts commit 15d91349a83253e4786ba96cbd7155a8db75ac5a. * code cleanup --------- Co-authored-by: Banks Nussman <115251059+bnussman-akamai@users.noreply.github.com> Co-authored-by: Banks Nussman Co-authored-by: Connie Liu <139280159+coliu-akamai@users.noreply.github.com> Co-authored-by: Nikhil Agrawal <165884194+nikhagra-akamai@users.noreply.github.com> --- .../pr-11395-added-1734381247482.md | 5 + .../components/DatePicker/DateTimePicker.tsx | 42 ++- .../DateTimeRangePicker.stories.tsx | 194 +++++++--- .../DatePicker/DateTimeRangePicker.test.tsx | 298 +++++++++++++-- .../DatePicker/DateTimeRangePicker.tsx | 353 +++++++++++++----- 5 files changed, 708 insertions(+), 184 deletions(-) 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)) diff --git a/packages/manager/src/components/DatePicker/DateTimePicker.tsx b/packages/manager/src/components/DatePicker/DateTimePicker.tsx index b503ba37674..86c66ee834a 100644 --- a/packages/manager/src/components/DatePicker/DateTimePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimePicker.tsx @@ -1,6 +1,7 @@ import { Divider } from '@linode/ui'; +import { InputAdornment, TextField } from '@linode/ui'; import { Box } from '@linode/ui'; -import { TextField } 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, @@ -75,6 +76,8 @@ export const DateTimePicker = ({ value = null, }: DateTimePickerProps) => { const [anchorEl, setAnchorEl] = useState(null); + + // Current and original states const [selectedDateTime, setSelectedDateTime] = useState( value ); @@ -82,6 +85,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 +125,8 @@ export const DateTimePicker = ({ const handleApply = () => { setAnchorEl(null); + setOriginalDateTime(selectedDateTime); + setOriginalTimeZone(selectedTimeZone); onChange(selectedDateTime); if (onApply) { @@ -124,6 +136,9 @@ export const DateTimePicker = ({ const handleClose = () => { setAnchorEl(null); + setSelectedDateTime(originalDateTime); + setSelectedTimeZone(originalTimeZone); + if (onCancel) { onCancel(); } @@ -139,6 +154,22 @@ export const DateTimePicker = ({ + + + ), + sx: { paddingLeft: '32px' }, + }} value={ selectedDateTime ? `${selectedDateTime.toFormat(format)}${ @@ -146,9 +177,9 @@ export const DateTimePicker = ({ }` : '' } - InputProps={{ readOnly: true }} errorText={errorText} label={label} + noMarginTop onClick={(event) => setAnchorEl(event.currentTarget)} placeholder={placeholder} /> @@ -247,10 +278,6 @@ export const DateTimePicker = ({ 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 df0314b6607..02bdd4f808b 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.test.tsx @@ -1,15 +1,45 @@ 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'; +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', () => { - 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(); }); @@ -21,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 @@ -29,12 +62,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: `${currentYear}-${currentMonth}-10T00:00:00.000-06:00`, + timeZone: null, + }); }); it('should show error when end date-time is before start date-time', async () => { @@ -50,24 +84,22 @@ 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(); }); 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'); @@ -88,18 +120,232 @@ 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'); + 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(); + }); + + it('should set the date range for the last 24 hours when the "Last 24 Hours" preset is selected', async () => { + renderWithTheme(); + + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "Last 24 Hours" option + const last24HoursOption = screen.getByText('Last 24 Hours'); + await userEvent.click(last24HoursOption); + + // 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.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 () => { + renderWithTheme(); + + // 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); + + // Expected start and end dates in ISO format + const expectedStartDateISO = fixedNow.minus({ days: 7 }).toISO(); + const expectedEndDateISO = fixedNow.toISO(); + + // Verify that onChange is called with the correct date range + expect(onChangeMock).toHaveBeenCalledWith({ + end: expectedEndDateISO, + preset: '7days', + start: expectedStartDateISO, + timeZone: null, + }); + expect( + 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 () => { + renderWithTheme(); + + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "Last 30 Days" option + const last30DaysOption = screen.getByText('Last 30 Days'); + await userEvent.click(last30DaysOption); + + // 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({ + end: expectedEndDateISO, + preset: '30days', + start: expectedStartDateISO, + timeZone: null, + }); + expect( + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); + }); + + it('should set the date range for this month when the "This Month" preset is selected', async () => { + renderWithTheme(); + + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "This Month" option + const thisMonthOption = screen.getByText('This Month'); + await userEvent.click(thisMonthOption); + + // 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({ + end: expectedEndDateISO, + preset: 'this_month', + start: expectedStartDateISO, + timeZone: null, + }); + expect( + screen.queryByRole('button', { name: 'Presets' }) + ).not.toBeInTheDocument(); + }); + + it('should set the date range for last month when the "Last Month" preset is selected', async () => { + renderWithTheme(); + + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "Last Month" option + const lastMonthOption = screen.getByText('Last Month'); + await userEvent.click(lastMonthOption); + + const lastMonth = fixedNow.minus({ months: 1 }); + + // 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({ + end: expectedEndDateISO, + preset: 'last_month', + start: expectedStartDateISO, + timeZone: null, + }); + expect( + 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 () => { + renderWithTheme(); + + // Open the presets dropdown + const presetsDropdown = screen.getByLabelText('Date Presets'); + await userEvent.click(presetsDropdown); + + // Select the "Custom Range" option + const customRange = screen.getByText('Custom'); + await userEvent.click(customRange); + + // 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(); + + // 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 66170f05586..7083ba7bee8 100644 --- a/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx +++ b/packages/manager/src/components/DatePicker/DateTimeRangePicker.tsx @@ -1,70 +1,141 @@ -import { Box } from '@linode/ui'; +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'; 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; + +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 */ - 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. - */ + + /** 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; + + /** 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 = ({ - 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) => { +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' }, + { label: 'This Month', value: 'this_month' }, + { label: 'Last Month', value: 'last_month' }, + { label: 'Custom', value: 'custom_range' }, +]; + +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 = { label: '', value: '' }, + 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 ); const [endDateTime, setEndDateTime] = useState( endDateTimeValue ); + const [presetValue, setPresetValue] = useState<{ + label: string; + value: string; + }>(presetsDefaultValue); const [startTimeZone, setStartTimeZone] = useState( startTimeZoneValue ); + const [startDateError, setStartDateError] = useState(null); + const [endDateError, setEndDateError] = useState(null); + const [showPresets, setShowPresets] = useState(enablePresets); - const [startDateError, setStartDateError] = useState(); - const [endDateError, setEndDateError] = useState(); + const theme = useTheme(); + const isSmallScreen = useMediaQuery(theme.breakpoints.down('md')); const validateDates = ( start: DateTime | null, @@ -73,72 +144,170 @@ export const DateTimeRangePicker = ({ ) => { if (start && end) { if (source === 'start' && start > end) { - setStartDateError('Start date/time cannot be after the end date/time.'); - setEndDateError(undefined); + setStartDateError(startDateErrorMessage); return; } if (source === 'end' && end < start) { - setEndDateError('End date/time cannot be before the start date/time.'); - setStartDateError(undefined); + setEndDateError(endDateErrorMessage); return; } } - // Reset validation errors if valid - setStartDateError(undefined); - setEndDateError(undefined); + // Reset validation errors + setStartDateError(null); + setEndDateError(null); + }; + + const handlePresetSelection = (value: DatePresetType) => { + 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); + setPresetValue( + presetsOptions.find((option) => option.value === value) ?? + presetsDefaultValue + ); + + if (onChange) { + onChange({ + end: newEndDateTime?.toISO() ?? null, + preset: value, + start: newStartDateTime?.toISO() ?? null, + timeZone: startTimeZone, + }); + } + + setShowPresets(value !== 'custom_range'); }; const handleStartDateTimeChange = (newStart: DateTime | null) => { setStartDateTime(newStart); validateDates(newStart, endDateTime, 'start'); - onChange(newStart, endDateTime, startTimeZone); + if (onChange) { + onChange({ + end: endDateTime?.toISO() ?? null, + preset: 'custom_range', + start: newStart?.toISO() ?? null, + timeZone: 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); + if (onChange) { + onChange({ + end: newEnd?.toISO() ?? null, + preset: 'custom_range', + start: startDateTime?.toISO() ?? null, + timeZone: startTimeZone, + }); + } }; return ( - - {/* Start DateTime Picker */} - - - {/* End DateTime Picker */} - + + {showPresets ? ( + { + if (selection) { + handlePresetSelection(selection.value as DatePresetType); + } + }} + defaultValue={presetsDefaultValue} + disableClearable + fullWidth + label={presetsLabel} + noMarginTop + options={presetsOptions} + placeholder={presetsPlaceholder} + value={presetValue} + /> + ) : ( + + setStartTimeZone(value), + value: startTimeZone, + }} + errorText={startDateError ?? undefined} + format={format} + label={startLabel} + onChange={handleStartDateTimeChange} + placeholder={startDatePlaceholder} + showTimeZone={showStartTimeZone} + timeSelectProps={{ label: 'Start Time' }} + value={startDateTime} + /> + + + { + setShowPresets(true); + setPresetValue(presetsDefaultValue); + }} + variant="text" + > + Presets + + + + )} ); };