5 changed files with 127 additions and 44 deletions
-
4package.json
-
57src/components/Calendar.tsx
-
39src/components/CalendarHeader.tsx
-
69src/components/DayColumn.tsx
-
2src/types.ts
@ -0,0 +1,39 @@ |
|||
import { format } from 'date-fns'; |
|||
|
|||
interface CalendarHeaderProps { |
|||
currentDate: Date; |
|||
onPrevWeek: () => void; |
|||
onNextWeek: () => void; |
|||
} |
|||
|
|||
function CalendarHeader({ currentDate, onPrevWeek, onNextWeek }: CalendarHeaderProps) { |
|||
return ( |
|||
<div className="flex items-center justify-center gap-4"> |
|||
<button |
|||
onClick={onPrevWeek} |
|||
className="p-2 rounded-full hover:bg-gray-100" |
|||
> |
|||
{/* Left arrow icon */} |
|||
<svg xmlns="http://www.w3.org/2000/svg" className="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor"> |
|||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 19l-7-7 7-7" /> |
|||
</svg> |
|||
</button> |
|||
|
|||
<h2 className="text-xl font-semibold"> |
|||
{format(currentDate, 'MMMM yyyy')} |
|||
</h2> |
|||
|
|||
<button |
|||
onClick={onNextWeek} |
|||
className="p-2 rounded-full hover:bg-gray-100" |
|||
> |
|||
{/* Right arrow icon */} |
|||
<svg xmlns="http://www.w3.org/2000/svg" className="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor"> |
|||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 5l7 7-7 7" /> |
|||
</svg> |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
|
|||
export default CalendarHeader; |
@ -0,0 +1,69 @@ |
|||
import React, { useRef } from 'react'; |
|||
import { motion } from 'framer-motion'; |
|||
import { format, addDays } from 'date-fns'; |
|||
import { DraggableEvent } from './DraggableEvent'; |
|||
|
|||
interface DayColumnProps { |
|||
date: Date; |
|||
events: Event[]; |
|||
isMobile: boolean; |
|||
index: number; |
|||
onEventClick: (event: Event) => void; |
|||
onDragStart: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void; |
|||
onDragEnd: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void; |
|||
handleEventMove: (eventId: string, newDate: Date) => void; |
|||
onDayChange: (dir: 'left' | 'right') => void; |
|||
isClient: boolean; |
|||
} |
|||
|
|||
const DayColumn: React.FC<DayColumnProps> = ({ |
|||
date, |
|||
events, |
|||
isMobile, |
|||
index, |
|||
onEventClick, |
|||
onDragStart, |
|||
onDragEnd, |
|||
handleEventMove, |
|||
onDayChange, |
|||
isClient |
|||
}) => { |
|||
const columnRef = useRef<HTMLDivElement>(null); |
|||
|
|||
const sortedEvents = [...events].sort((a, b) => a.start.getTime() - b.start.getTime()); |
|||
|
|||
return ( |
|||
<motion.div |
|||
ref={columnRef} |
|||
className={`flex-1 ${isMobile ? 'min-w-[calc(100vw-32px)]' : ''} bg-gray-50 rounded-lg p-2 relative`} |
|||
data-day={index} |
|||
> |
|||
<div className="font-bold mb-2 text-black text-xl"> |
|||
{format(date, 'EEE, MMM d')} |
|||
</div> |
|||
|
|||
<div className="relative z-10"> |
|||
{sortedEvents |
|||
.filter(event => event && event.id) |
|||
.map((event, index) => ( |
|||
<DraggableEvent |
|||
key={event.id} |
|||
event={event} |
|||
date={format(date, 'yyyy-MM-dd')} |
|||
onEventClick={onEventClick} |
|||
onDragStart={onDragStart} |
|||
onDragEnd={onDragEnd} |
|||
onDayChange={(dir) => { |
|||
onDayChange(dir); |
|||
const newDate = addDays(date, dir === 'left' ? -1 : 1); |
|||
handleEventMove(event.id, newDate); |
|||
}} |
|||
isClient={isClient} |
|||
/> |
|||
))} |
|||
</div> |
|||
</motion.div> |
|||
); |
|||
}; |
|||
|
|||
export default DayColumn; |
Write
Preview
Loading…
Cancel
Save
Reference in new issue