Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Work on Formatted Dates #1424

Merged
merged 4 commits into from
Feb 21, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 5 additions & 3 deletions src/components/FieldOverviewCard/FieldOverviewCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import { To } from 'history';
import { DateTime } from 'luxon';
import { ReactNode } from 'react';
import { makeStyles } from 'tss-react/mui';
import { useDateTimeFormatter } from '../Formatters';
import { FormattedDateTime } from '../Formatters';
import { HugeIcon, HugeIconProps } from '../Icons';
import { ButtonLink, CardActionAreaLink } from '../Routing';

Expand Down Expand Up @@ -83,7 +83,6 @@ export const FieldOverviewCard = ({
viewLabel: buttonLabel,
}: FieldOverviewCardProps) => {
const { classes, cx } = useStyles();
const dateTimeFormatter = useDateTimeFormatter();

const showData = !loading && !redacted;
const ActionArea = showData && data?.to ? CardActionAreaLink : CardActionArea;
Expand Down Expand Up @@ -149,7 +148,10 @@ export const FieldOverviewCard = ({
{loading ? (
<Skeleton />
) : data?.updatedAt ? (
<> Updated {dateTimeFormatter(data.updatedAt)}</>
<>
{' '}
Updated <FormattedDateTime date={data.updatedAt} />
</>
) : null}
</Typography>
)}
Expand Down
35 changes: 30 additions & 5 deletions src/components/Formatters/FormattedDate.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,20 +8,22 @@ import { CalendarDate, Nullable } from '~/common';
import { useLocale } from '../../hooks';
import { useDateFormatter, useDateTimeFormatter } from './useDateFormatter';

export const FormattedDate = ({
export const FormattedDate = memo(function FormattedDate({
date,
displayOptions,
}: {
date: Nullable<CalendarDate>;
displayOptions?: DateTimeFormatOptions;
}) => {
}) {
date = asLuxonInstance(date, CalendarDate);

const format = useDateFormatter();
return date ? (
<Tooltip title={format(date, DateTime.DATE_HUGE)}>
<time dateTime={date.toISODate()}>{format(date, displayOptions)}</time>
</Tooltip>
) : null;
};
});

export const FormattedDateRange = ({
start,
Expand Down Expand Up @@ -55,20 +57,28 @@ FormattedDateRange.orNull = (range: Nullable<DateRange>) =>
<FormattedDateRange range={range} />
);

export const FormattedDateTime = ({ date }: { date: Nullable<DateTime> }) => {
export const FormattedDateTime = memo(function FormattedDateTime({
date,
}: {
date: Nullable<DateTime>;
}) {
date = asLuxonInstance(date, DateTime);

const format = useDateTimeFormatter();
return date ? (
<Tooltip title={format(date, DateTime.DATETIME_HUGE)}>
<time dateTime={date.toUTC().toISO()}>{format(date)}</time>
</Tooltip>
) : null;
};
});

export const RelativeDateTime = memo(function RelativeDateTime({
date,
}: {
date: DateTime;
}) {
date = asLuxonInstance(date, DateTime)!;

const locale = useLocale();
date = locale ? date.setLocale(locale) : date;
const absoluteFormat = useDateTimeFormatter();
Expand Down Expand Up @@ -102,3 +112,18 @@ export const RelativeDateTime = memo(function RelativeDateTime({
</Tooltip>
);
});

// Under some circumstances, the Apollo scalar read policy is ignored, causing
// this date to just be an ISO string.
// This is just a workaround, to try to prevent errors for users.
// https://github.com/apollographql/apollo-client/issues/9293
function asLuxonInstance(
date: DateTime | null | undefined,
cls: typeof DateTime
) {
return !date
? null
: cls.isDateTime(date)
? date
: cls.fromISO(date as unknown as string);
}
8 changes: 6 additions & 2 deletions src/components/Formatters/useDateFormatter.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,9 @@ const useTimezone = () => {
return timezone;
};

// Returns function for format date or date range
/**
* @deprecated Use FormattedDate component instead
*/
export const useDateFormatter = () => {
const locale = useLocale();

Expand All @@ -27,7 +29,9 @@ export const useDateFormatter = () => {
return formatDate;
};

// Returns function for format date time or date time range
/**
* @deprecated Use FormattedDateTime component instead
*/
export const useDateTimeFormatter = () => {
const locale = useLocale();
const timeZone = useTimezone();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -93,9 +93,11 @@ export const Due = ({ date }: { date: CalendarDate }) => (
date={date}
displayOptions={
date.diffNow('years').years < 1 // same year
? omit(DateTime.DATE_SHORT, 'year')
? DATE_SHORT_NO_YEAR
: undefined
}
/>
</>
);

const DATE_SHORT_NO_YEAR = omit(DateTime.DATE_SHORT, 'year');
9 changes: 4 additions & 5 deletions src/components/ProjectMemberCard/ProjectMemberCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import { makeStyles } from 'tss-react/mui';
import { RoleLabels } from '~/api/schema.graphql';
import { labelsFrom } from '~/common';
import { Avatar } from '../Avatar';
import { useDateTimeFormatter } from '../Formatters';
import { FormattedDateTime } from '../Formatters';
import { ProjectMemberCardFragment } from './ProjectMember.graphql';

const useStyles = makeStyles()(({ spacing }) => ({
Expand Down Expand Up @@ -47,9 +47,6 @@ export const ProjectMemberCard = ({
className,
}: ProjectMemberCardProps) => {
const { classes } = useStyles();
const dateTimeFormatter = useDateTimeFormatter();

const createdAtString = dateTimeFormatter(projectMember?.createdAt);

return (
<Card className={className}>
Expand Down Expand Up @@ -94,7 +91,9 @@ export const ProjectMemberCard = ({
{!projectMember ? (
<Skeleton variant="text" width="23ch" />
) : (
`Member Since ${createdAtString}`
<>
Member Since <FormattedDateTime date={projectMember.createdAt} />
</>
)}
</Typography>
</CardActions>
Expand Down
12 changes: 7 additions & 5 deletions src/components/files/FileVersionItem/FileVersionItem.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import {
ListItemText,
} from '@mui/material';
import { makeStyles } from 'tss-react/mui';
import { useDateTimeFormatter } from '../../Formatters';
import { FormattedDateTime } from '../../Formatters';
import {
FileActionsPopup as ActionsMenu,
FileAction,
Expand Down Expand Up @@ -35,7 +35,6 @@ interface FileVersionItemProps {

export const FileVersionItem = (props: FileVersionItemProps) => {
const { classes } = useStyles();
const formatDate = useDateTimeFormatter();
const { openFilePreview } = useFileActions();
const { version, actions } = props;

Expand Down Expand Up @@ -72,9 +71,12 @@ export const FileVersionItem = (props: FileVersionItemProps) => {
onClick={() => openFilePreview(version)}
className={classes.text}
primary={name}
secondary={`Created on ${formatDate(createdAt)}${
createdByUser ? ` by ${createdByUser}` : ''
}`}
secondary={
<>
Created <FormattedDateTime date={createdAt} />
{createdByUser ? ` by ${createdByUser}` : ''}
</>
}
/>
<ListItemSecondaryAction>
<ActionsMenu item={version} actions={menuActions} />
Expand Down
9 changes: 6 additions & 3 deletions src/scenes/Engagement/CeremonyCard/CeremonyCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import { canEditAny } from '~/common';
import { useDialog } from '../../../components/Dialog';
import { DialogForm } from '../../../components/Dialog/DialogForm';
import { DateField, FieldGroup, SubmitError } from '../../../components/form';
import { useDateFormatter } from '../../../components/Formatters';
import { FormattedDate } from '../../../components/Formatters';
import { Redacted } from '../../../components/Redacted';
import {
CeremonyCardFragment,
Expand Down Expand Up @@ -60,7 +60,6 @@ export const CeremonyCard = ({
const loading = canRead == null;

const { classes, cx } = useStyles();
const formatDate = useDateFormatter();
const [updateCeremony] = useMutation(UpdateCeremonyDocument);
const [dialogState, openDialog] = useDialog();

Expand Down Expand Up @@ -137,7 +136,11 @@ export const CeremonyCard = ({
>
{!loading ? (
canRead && ceremony?.estimatedDate.canRead ? (
formatDate(estimatedDate?.value) || <>&nbsp;</>
estimatedDate?.value ? (
<FormattedDate date={estimatedDate.value} />
) : (
<>&nbsp;</>
)
) : (
<Redacted
info="You don't have permission to view the estimated date"
Expand Down
9 changes: 5 additions & 4 deletions src/scenes/Engagement/CeremonyCard/LargeDate.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { Skeleton, Typography } from '@mui/material';
import { useState } from 'react';
import { makeStyles } from 'tss-react/mui';
import { CalendarDate, Nullable, SecuredProp } from '~/common';
import { useDateFormatter } from '../../../components/Formatters';
import { FormattedDate } from '../../../components/Formatters';
import { Redacted } from '../../../components/Redacted';

const useStyles = makeStyles()(({ palette, spacing }) => ({
Expand Down Expand Up @@ -38,7 +38,6 @@ interface LargeDateProps {

export const LargeDate = ({ date, className }: LargeDateProps) => {
const { classes, cx } = useStyles();
const formatDate = useDateFormatter();
const [placeholderNow] = useState(() => CalendarDate.local());

return (
Expand All @@ -52,10 +51,12 @@ export const LargeDate = ({ date, className }: LargeDateProps) => {
)}
>
{date?.value ? (
formatDate(date.value)
<FormattedDate date={date.value} />
) : (
<>
<span className={classes.hidden}>{formatDate(placeholderNow)}</span>
<span className={classes.hidden}>
<FormattedDate date={placeholderNow} />
</span>
{!date ? (
<Skeleton variant="rectangular" className={classes.skeleton} />
) : !date.canRead ? (
Expand Down
5 changes: 2 additions & 3 deletions src/scenes/Languages/Detail/LanguageDetail.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import {
import { Error } from '../../../components/Error';
import { Fab } from '../../../components/Fab';
import {
useDateFormatter,
FormattedDate,
useNumberFormatter,
} from '../../../components/Formatters';
import { IconButton } from '../../../components/IconButton';
Expand Down Expand Up @@ -101,7 +101,6 @@ export const LanguageDetail = () => {

const canEditAnyFields = canEditAny(language) || canEditAny(ethnologue);

const formatDate = useDateFormatter();
const formatNumber = useNumberFormatter();

const [removeLocation, { loading: removing }] = useMutation(
Expand Down Expand Up @@ -216,7 +215,7 @@ export const LanguageDetail = () => {
/>
<DisplayProperty
label="Sponsor Start Date"
value={formatDate(sponsorStartDate?.value)}
value={<FormattedDate date={sponsorStartDate?.value} />}
loading={!language}
/>
<DisplayProperty
Expand Down
5 changes: 2 additions & 3 deletions src/scenes/Partners/Detail/PartnerDetail.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ import { BooleanProperty } from '../../../components/BooleanProperty';
import { DataButton } from '../../../components/DataButton';
import { useDialog } from '../../../components/Dialog';
import { Error } from '../../../components/Error';
import { useDateTimeFormatter } from '../../../components/Formatters';
import { FormattedDateTime } from '../../../components/Formatters';
import { IconButton } from '../../../components/IconButton';
import { ProjectListItemCard } from '../../../components/ProjectListItemCard';
import { TogglePinButton } from '../../../components/TogglePinButton';
Expand Down Expand Up @@ -98,7 +98,6 @@ const useStyles = makeStyles()(({ spacing, breakpoints, palette }) => ({
export const PartnerDetail = () => {
const { classes } = useStyles();
const { partnerId = '' } = useParams();
const formatDateTime = useDateTimeFormatter();

const { data, error } = useQuery(PartnerDocument, {
variables: {
Expand Down Expand Up @@ -158,7 +157,7 @@ export const PartnerDetail = () => {
</Typography>
{partner && (
<Typography variant="body2" color="textSecondary">
Created {formatDateTime(partner.createdAt)}
Created <FormattedDateTime date={partner.createdAt} />
</Typography>
)}
</div>
Expand Down
5 changes: 2 additions & 3 deletions src/scenes/Projects/Files/ProjectFilesList.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,8 @@ import {
import { fileIcon } from '../../../components/files/fileTypes';
import {
formatFileSize,
FormattedDateTime,
parseFileNameAndExtension,
useDateTimeFormatter,
} from '../../../components/Formatters';
import { IconButton } from '../../../components/IconButton';
import { ContentContainer } from '../../../components/Layout';
Expand Down Expand Up @@ -74,7 +74,6 @@ const ProjectFilesListWrapped = () => {
const { classes } = useStyles();
const navigate = useNavigate();
const { projectUrl } = useProjectId();
const formatDate = useDateTimeFormatter();

const { openFilePreview } = useFileActions();

Expand Down Expand Up @@ -151,7 +150,7 @@ const ProjectFilesListWrapped = () => {
width: 150,
valueGetter: ({ row }) =>
row.__typename === 'File' ? row.modifiedAt : row.createdAt,
valueFormatter: ({ value }) => formatDate(value),
renderCell: ({ value }) => <FormattedDateTime date={value} />,
},
{
headerName: 'Modified By',
Expand Down