mirror of
https://github.com/kennethnym/aris.git
synced 2026-03-20 09:01:19 +00:00
feat(caldav): add FeedItemRenderer (#74)
Implement renderCalDavFeedItem using JRX JSX to render CalDAV events as FeedCard components. Bump @nym.sh/jrx to 0.2.0 for null/undefined child support. Co-authored-by: Ona <no-reply@ona.com>
This commit is contained in:
@@ -9,6 +9,7 @@
|
||||
"test:live": "bun run scripts/test-live.ts"
|
||||
},
|
||||
"dependencies": {
|
||||
"@aelis/components": "workspace:*",
|
||||
"@aelis/core": "workspace:*",
|
||||
"ical.js": "^2.1.0",
|
||||
"tsdav": "^2.1.7"
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
export { CalDavCalendarKey, type CalendarContext } from "./calendar-context.ts"
|
||||
export { CalDavSource, type CalDavSourceOptions } from "./caldav-source.ts"
|
||||
export { parseICalEvents, type ICalTimeRange } from "./ical-parser.ts"
|
||||
export { renderCalDavFeedItem } from "./renderer.tsx"
|
||||
export {
|
||||
AttendeeRole,
|
||||
AttendeeStatus,
|
||||
|
||||
74
packages/aelis-source-caldav/src/renderer.tsx
Normal file
74
packages/aelis-source-caldav/src/renderer.tsx
Normal file
@@ -0,0 +1,74 @@
|
||||
/** @jsxImportSource @nym.sh/jrx */
|
||||
|
||||
import type { FeedItemRenderer } from "@aelis/core"
|
||||
|
||||
import { FeedCard, SansSerifText, SerifText } from "@aelis/components"
|
||||
|
||||
import type { CalDavEventData } from "./types.ts"
|
||||
|
||||
import { CalDavEventStatus } from "./types.ts"
|
||||
|
||||
function formatTime(date: Date): string {
|
||||
const hours = date.getHours()
|
||||
const minutes = date.getMinutes()
|
||||
const period = hours >= 12 ? "PM" : "AM"
|
||||
const h = hours % 12 || 12
|
||||
const m = minutes.toString().padStart(2, "0")
|
||||
return `${h}:${m} ${period}`
|
||||
}
|
||||
|
||||
function formatTimeRange(data: CalDavEventData): string {
|
||||
if (data.isAllDay) {
|
||||
return "All day"
|
||||
}
|
||||
return `${formatTime(data.startDate)} – ${formatTime(data.endDate)}`
|
||||
}
|
||||
|
||||
function formatStatus(status: CalDavEventData["status"]): string | null {
|
||||
if (status === CalDavEventStatus.Cancelled) return "Cancelled"
|
||||
if (status === CalDavEventStatus.Tentative) return "Tentative"
|
||||
return null
|
||||
}
|
||||
|
||||
export const renderCalDavFeedItem: FeedItemRenderer<"caldav-event", CalDavEventData> = (item) => {
|
||||
const { data, slots } = item
|
||||
const statusLabel = formatStatus(data.status)
|
||||
const attendeeCount = data.attendees.length
|
||||
|
||||
return (
|
||||
<FeedCard>
|
||||
{statusLabel ? <SansSerifText content={statusLabel} style="text-xs uppercase" /> : null}
|
||||
|
||||
<SerifText content={data.title} style="text-lg" />
|
||||
|
||||
<SansSerifText content={formatTimeRange(data)} style="text-sm" />
|
||||
|
||||
{data.calendarName ? (
|
||||
<SansSerifText content={data.calendarName} style="text-sm text-secondary" />
|
||||
) : null}
|
||||
|
||||
{data.location ? (
|
||||
<SansSerifText content={data.location} style="text-sm text-secondary" />
|
||||
) : null}
|
||||
|
||||
{attendeeCount > 0 ? (
|
||||
<SansSerifText
|
||||
content={`${attendeeCount} attendee${attendeeCount === 1 ? "" : "s"}`}
|
||||
style="text-sm text-secondary"
|
||||
/>
|
||||
) : null}
|
||||
|
||||
{slots?.insight?.content ? (
|
||||
<SansSerifText content={slots.insight.content} style="text-sm" />
|
||||
) : null}
|
||||
|
||||
{slots?.preparation?.content ? (
|
||||
<SansSerifText content={slots.preparation.content} style="text-sm" />
|
||||
) : null}
|
||||
|
||||
{slots?.crossSource?.content ? (
|
||||
<SansSerifText content={slots.crossSource.content} style="text-sm" />
|
||||
) : null}
|
||||
</FeedCard>
|
||||
)
|
||||
}
|
||||
7
packages/aelis-source-caldav/tsconfig.json
Normal file
7
packages/aelis-source-caldav/tsconfig.json
Normal file
@@ -0,0 +1,7 @@
|
||||
{
|
||||
"extends": "../../tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"jsxImportSource": "@nym.sh/jrx"
|
||||
},
|
||||
"include": ["src"]
|
||||
}
|
||||
Reference in New Issue
Block a user