feat(events): render public detail page in event timezone

Event detail page formatted dates in viewer-local time, so a Lisbon
viewer of a Toronto ET event saw the time in WEST. Format with
event.displayTimezone instead so attendees see the event's intended
wall-clock + zone suffix ("6:00 AM EDT") regardless of where they sit.

useEventDateUtils.formatDate / formatTime / formatDateRange / isToday
now accept a { timeZone } option and pass it to Intl.DateTimeFormat.
Existing call sites that don't pass timeZone fall through to viewer-
local, matching prior behaviour.
This commit is contained in:
Jennie Robinson Faber 2026-05-19 10:45:11 +01:00
parent a76ba2f8c7
commit acbd3c0737
2 changed files with 56 additions and 35 deletions

View file

@ -1,85 +1,98 @@
// Utility composable for event date handling with timezone support // Utility composable for event date handling with timezone support.
// Pass `{ timeZone: event.displayTimezone }` to render in the event's TZ.
export const useEventDateUtils = () => { export const useEventDateUtils = () => {
const TIMEZONE = "America/Toronto"; const DEFAULT_TIMEZONE = "America/Toronto";
// Format a date to a specific format
const formatDate = (date, options = {}) => { const formatDate = (date, options = {}) => {
if (!date) return "";
const dateObj = date instanceof Date ? date : new Date(date); const dateObj = date instanceof Date ? date : new Date(date);
const { month = "short", day = "numeric", year = "numeric" } = options; if (isNaN(dateObj.getTime())) return "";
const {
month = "short",
day = "numeric",
year = "numeric",
weekday,
timeZone,
} = options;
return new Intl.DateTimeFormat("en-US", { return new Intl.DateTimeFormat("en-US", {
...(weekday && { weekday }),
month, month,
day, day,
year, year,
...(timeZone && { timeZone }),
}).format(dateObj); }).format(dateObj);
}; };
// Format event date range const formatDateRange = (startDate, endDate, compact = false, timeZone) => {
const formatDateRange = (startDate, endDate, compact = false) => {
if (!startDate || !endDate) return "No dates"; if (!startDate || !endDate) return "No dates";
const start = new Date(startDate); const start = new Date(startDate);
const end = new Date(endDate); const end = new Date(endDate);
const startMonth = start.toLocaleDateString("en-US", { month: "short" }); const tzOpts = timeZone ? { timeZone } : {};
const endMonth = end.toLocaleDateString("en-US", { month: "short" }); const startMonth = start.toLocaleDateString("en-US", { month: "short", ...tzOpts });
const startDay = start.getDate(); const endMonth = end.toLocaleDateString("en-US", { month: "short", ...tzOpts });
const endDay = end.getDate(); const startDay = Number(
const year = end.getFullYear(); start.toLocaleDateString("en-US", { day: "numeric", ...tzOpts }),
);
const endDay = Number(
end.toLocaleDateString("en-US", { day: "numeric", ...tzOpts }),
);
const year = Number(
end.toLocaleDateString("en-US", { year: "numeric", ...tzOpts }),
);
const startMonthIdx = startMonth; // compared as label string
const endMonthIdx = endMonth;
const startYear = Number(
start.toLocaleDateString("en-US", { year: "numeric", ...tzOpts }),
);
if (compact) { if (compact) {
if ( if (startMonthIdx === endMonthIdx && startYear === year) {
start.getMonth() === end.getMonth() &&
start.getFullYear() === end.getFullYear()
) {
return `${startMonth} ${startDay}-${endDay}`; return `${startMonth} ${startDay}-${endDay}`;
} }
return `${startMonth} ${startDay} - ${endMonth} ${endDay}`; return `${startMonth} ${startDay} - ${endMonth} ${endDay}`;
} }
if ( if (startMonthIdx === endMonthIdx && startYear === year) {
start.getMonth() === end.getMonth() &&
start.getFullYear() === end.getFullYear()
) {
return `${startMonth} ${startDay}-${endDay}, ${year}`; return `${startMonth} ${startDay}-${endDay}, ${year}`;
} else if (start.getFullYear() === end.getFullYear()) { } else if (startYear === year) {
return `${startMonth} ${startDay} - ${endMonth} ${endDay}, ${year}`; return `${startMonth} ${startDay} - ${endMonth} ${endDay}, ${year}`;
} else { } else {
return `${formatDate(startDate)} - ${formatDate(endDate)}`; return `${formatDate(startDate, { timeZone })} - ${formatDate(endDate, { timeZone })}`;
} }
}; };
// Check if a date is in the past
const isPastDate = (date) => { const isPastDate = (date) => {
const dateObj = date instanceof Date ? date : new Date(date); const dateObj = date instanceof Date ? date : new Date(date);
const now = new Date(); return dateObj < new Date();
return dateObj < now;
}; };
// Check if a date is today const isToday = (date, timeZone) => {
const isToday = (date) => {
const dateObj = date instanceof Date ? date : new Date(date); const dateObj = date instanceof Date ? date : new Date(date);
const today = new Date(); const today = new Date();
const opts = { year: "numeric", month: "2-digit", day: "2-digit", ...(timeZone && { timeZone }) };
return ( return (
dateObj.getDate() === today.getDate() && dateObj.toLocaleDateString("en-US", opts) ===
dateObj.getMonth() === today.getMonth() && today.toLocaleDateString("en-US", opts)
dateObj.getFullYear() === today.getFullYear()
); );
}; };
// Get a readable time string const formatTime = (date, includeSeconds = false, timeZone) => {
const formatTime = (date, includeSeconds = false) => {
const dateObj = date instanceof Date ? date : new Date(date); const dateObj = date instanceof Date ? date : new Date(date);
const options = { return new Intl.DateTimeFormat("en-US", {
hour: "2-digit", hour: "2-digit",
minute: "2-digit", minute: "2-digit",
...(includeSeconds && { second: "2-digit" }), ...(includeSeconds && { second: "2-digit" }),
}; ...(timeZone && { timeZone }),
return new Intl.DateTimeFormat("en-US", options).format(dateObj); }).format(dateObj);
}; };
return { return {
TIMEZONE, DEFAULT_TIMEZONE,
// Legacy alias for callers that hard-coded the constant.
TIMEZONE: DEFAULT_TIMEZONE,
formatDate, formatDate,
formatDateRange, formatDateRange,
isPastDate, isPastDate,

View file

@ -197,21 +197,29 @@ onMounted(async () => {
} }
}); });
const eventTimeZone = computed(
() => event.value?.displayTimezone || "America/Toronto",
);
const formatDate = (dateStr) => { const formatDate = (dateStr) => {
if (!dateStr) return "";
const d = new Date(dateStr); const d = new Date(dateStr);
return new Intl.DateTimeFormat("en-US", { return new Intl.DateTimeFormat("en-US", {
weekday: "long", weekday: "long",
year: "numeric", year: "numeric",
month: "long", month: "long",
day: "numeric", day: "numeric",
timeZone: eventTimeZone.value,
}).format(d); }).format(d);
}; };
const formatTime = (start, end) => { const formatTime = (start, end) => {
if (!start || !end) return "";
const fmt = new Intl.DateTimeFormat("en-US", { const fmt = new Intl.DateTimeFormat("en-US", {
hour: "numeric", hour: "numeric",
minute: "2-digit", minute: "2-digit",
timeZoneName: "short", timeZoneName: "short",
timeZone: eventTimeZone.value,
}); });
return `${fmt.format(new Date(start))} ${fmt.format(new Date(end))}`; return `${fmt.format(new Date(start))} ${fmt.format(new Date(end))}`;
}; };