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