From e34d85946e193a728367ee7c5cdf24fa9b583806 Mon Sep 17 00:00:00 2001 From: Tom Smith Date: Fri, 25 Oct 2024 12:19:05 +0100 Subject: [PATCH] feat: add preview component (#22) --- src/components/RecurringDatesPreview.tsx | 55 ++++++++++++++++++++++++ src/schema/recurringDates.tsx | 11 +++++ src/types.ts | 6 +-- 3 files changed, 69 insertions(+), 3 deletions(-) create mode 100644 src/components/RecurringDatesPreview.tsx diff --git a/src/components/RecurringDatesPreview.tsx b/src/components/RecurringDatesPreview.tsx new file mode 100644 index 0000000..1779668 --- /dev/null +++ b/src/components/RecurringDatesPreview.tsx @@ -0,0 +1,55 @@ +import {DEFAULT_DATE_FORMAT, DEFAULT_TIME_FORMAT, format} from '@sanity/util/legacyDateFormat' +import {upperFirst} from 'lodash' +import React from 'react' +import {rrulestr} from 'rrule' +import type {ObjectSchemaType, PreviewProps} from 'sanity' + +import type {PluginConfig, RecurringDate, WithRequiredProperty} from '../types' + +type CastPreviewProps = PreviewProps & + RecurringDate & { + pluginConfig: WithRequiredProperty + } + +type RecurringDateObjectSchemaType = Omit & { + options?: PluginConfig +} + +export function RecurringDatesPreview(props: CastPreviewProps): React.JSX.Element { + const {startDate, endDate, rrule, schemaType, pluginConfig} = props + const options: RecurringDateObjectSchemaType = schemaType?.options + + const {dateTimeOptions, dateOnly} = { + ...pluginConfig, + ...options, + } + + const rule = rrule && rrulestr(rrule) + + const dateFormat = dateTimeOptions?.dateFormat || DEFAULT_DATE_FORMAT + const timeFormat = dateTimeOptions?.timeFormat || DEFAULT_TIME_FORMAT + + const start = startDate ? new Date(startDate) : undefined + const end = endDate ? new Date(endDate) : undefined + const sameDay = start && end && start.toDateString() === end.toDateString() + + let title = 'No start date' + if (dateOnly) { + title = start ? format(start, dateFormat) : 'No start date' + if (end && !sameDay) { + title += ` - ${format(end, dateFormat)}` + } + } else { + title = start ? format(start, `${dateFormat} ${timeFormat}`) : 'No start date' + if (end) { + title += ` - ${format(end, sameDay ? timeFormat : `${dateFormat} ${timeFormat}`)}` + } + } + + const previewProps = { + title, + subtitle: rule && upperFirst(rule.toText()), + } + + return props.renderDefault({...previewProps, ...props}) +} diff --git a/src/schema/recurringDates.tsx b/src/schema/recurringDates.tsx index 099b445..0974c5a 100644 --- a/src/schema/recurringDates.tsx +++ b/src/schema/recurringDates.tsx @@ -1,6 +1,8 @@ +import {CalendarIcon} from '@sanity/icons' import {defineField, SchemaTypeDefinition} from 'sanity' import {RecurringDates} from '../components/RecurringDate' +import {RecurringDatesPreview} from '../components/RecurringDatesPreview' import {PluginConfig, WithRequiredProperty} from '../types' export default ( @@ -12,6 +14,7 @@ export default ( name: 'recurringDates', title: 'Dates', type: 'object', + icon: CalendarIcon, fields: [ defineField({ title: 'Start Date', @@ -43,6 +46,14 @@ export default ( ], components: { input: (props) => RecurringDates({...props, pluginConfig: config}), + preview: (props) => RecurringDatesPreview({...props, pluginConfig: config}), + }, + preview: { + select: { + startDate: 'startDate', + endDate: 'endDate', + rrule: 'rrule', + }, }, }) } diff --git a/src/types.ts b/src/types.ts index 7a7f361..1646861 100644 --- a/src/types.ts +++ b/src/types.ts @@ -32,7 +32,7 @@ declare module 'sanity' { } export interface RecurringDate { - rrule: string - startDate: string - endDate: string + rrule?: string + startDate?: string + endDate?: string }