gpt-engineer-app[bot] commited on
Commit
b37da4d
·
1 Parent(s): 78dc268

Add "Add to Calendar" button

Browse files

Adds a button to the conference detail view allowing users to add the conference to their calendar (Google Calendar, Apple Calendar, etc.).

Files changed (1) hide show
  1. src/components/ConferenceDialog.tsx +86 -8
src/components/ConferenceDialog.tsx CHANGED
@@ -5,9 +5,16 @@ import {
5
  DialogHeader,
6
  DialogTitle,
7
  } from "@/components/ui/dialog";
8
- import { CalendarDays, Globe, Tag, Clock, AlarmClock } from "lucide-react";
9
  import { Conference } from "@/types/conference";
10
- import { formatDistanceToNow, parseISO, isValid } from "date-fns";
 
 
 
 
 
 
 
11
 
12
  interface ConferenceDialogProps {
13
  conference: Conference;
@@ -27,16 +34,87 @@ const ConferenceDialog = ({ conference, open, onOpenChange }: ConferenceDialogPr
27
  return "text-green-600";
28
  };
29
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
30
  return (
31
  <Dialog open={open} onOpenChange={onOpenChange}>
32
  <DialogContent className="dialog-content max-w-md">
33
  <DialogHeader>
34
- <DialogTitle className="text-xl font-bold">
35
- {conference.title}
36
- </DialogTitle>
37
- {conference.full_name && (
38
- <p className="text-sm text-neutral-600">{conference.full_name}</p>
39
- )}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
40
  </DialogHeader>
41
 
42
  <div className="space-y-4 py-4">
 
5
  DialogHeader,
6
  DialogTitle,
7
  } from "@/components/ui/dialog";
8
+ import { CalendarDays, Globe, Tag, Clock, AlarmClock, CalendarPlus } from "lucide-react";
9
  import { Conference } from "@/types/conference";
10
+ import { formatDistanceToNow, parseISO, isValid, format } from "date-fns";
11
+ import { Button } from "@/components/ui/button";
12
+ import {
13
+ DropdownMenu,
14
+ DropdownMenuContent,
15
+ DropdownMenuItem,
16
+ DropdownMenuTrigger,
17
+ } from "@/components/ui/dropdown-menu";
18
 
19
  interface ConferenceDialogProps {
20
  conference: Conference;
 
34
  return "text-green-600";
35
  };
36
 
37
+ const createCalendarEvent = (type: 'google' | 'apple') => {
38
+ const startDate = conference.start
39
+ ? parseISO(conference.start)
40
+ : parseISO(conference.date.split('-')[0].trim());
41
+ const endDate = conference.end
42
+ ? parseISO(conference.end)
43
+ : parseISO(conference.date.split('-')[1]?.trim() || conference.date);
44
+
45
+ const formatDateForGoogle = (date: Date) => format(date, "yyyyMMdd'T'HHmmss'Z'");
46
+ const formatDateForApple = (date: Date) => format(date, "yyyyMMdd'T'HHmmss");
47
+
48
+ const title = encodeURIComponent(conference.title);
49
+ const location = encodeURIComponent(conference.place);
50
+ const description = encodeURIComponent(
51
+ `Conference: ${conference.full_name || conference.title}\n` +
52
+ `Location: ${conference.place}\n` +
53
+ `Deadline: ${conference.deadline}\n` +
54
+ (conference.abstract_deadline ? `Abstract Deadline: ${conference.abstract_deadline}\n` : '') +
55
+ (conference.link ? `Website: ${conference.link}` : '')
56
+ );
57
+
58
+ if (type === 'google') {
59
+ const url = `https://calendar.google.com/calendar/render?action=TEMPLATE` +
60
+ `&text=${title}` +
61
+ `&dates=${formatDateForGoogle(startDate)}/${formatDateForGoogle(endDate)}` +
62
+ `&details=${description}` +
63
+ `&location=${location}` +
64
+ `&sprop=website:${encodeURIComponent(conference.link || '')}`;
65
+ window.open(url, '_blank');
66
+ } else {
67
+ const url = `data:text/calendar;charset=utf8,BEGIN:VCALENDAR
68
+ VERSION:2.0
69
+ BEGIN:VEVENT
70
+ URL:${conference.link || ''}
71
+ DTSTART:${formatDateForApple(startDate)}
72
+ DTEND:${formatDateForApple(endDate)}
73
+ SUMMARY:${title}
74
+ DESCRIPTION:${description}
75
+ LOCATION:${location}
76
+ END:VEVENT
77
+ END:VCALENDAR`;
78
+
79
+ const link = document.createElement('a');
80
+ link.href = url;
81
+ link.download = `${conference.title.toLowerCase().replace(/\s+/g, '-')}.ics`;
82
+ document.body.appendChild(link);
83
+ link.click();
84
+ document.body.removeChild(link);
85
+ }
86
+ };
87
+
88
  return (
89
  <Dialog open={open} onOpenChange={onOpenChange}>
90
  <DialogContent className="dialog-content max-w-md">
91
  <DialogHeader>
92
+ <div className="flex justify-between items-start">
93
+ <div>
94
+ <DialogTitle className="text-xl font-bold">
95
+ {conference.title}
96
+ </DialogTitle>
97
+ {conference.full_name && (
98
+ <p className="text-sm text-neutral-600">{conference.full_name}</p>
99
+ )}
100
+ </div>
101
+ <DropdownMenu>
102
+ <DropdownMenuTrigger asChild>
103
+ <Button variant="outline" size="sm">
104
+ <CalendarPlus className="h-4 w-4 mr-2" />
105
+ Add to Calendar
106
+ </Button>
107
+ </DropdownMenuTrigger>
108
+ <DropdownMenuContent align="end">
109
+ <DropdownMenuItem onClick={() => createCalendarEvent('google')}>
110
+ Add to Google Calendar
111
+ </DropdownMenuItem>
112
+ <DropdownMenuItem onClick={() => createCalendarEvent('apple')}>
113
+ Add to Apple Calendar
114
+ </DropdownMenuItem>
115
+ </DropdownMenuContent>
116
+ </DropdownMenu>
117
+ </div>
118
  </DialogHeader>
119
 
120
  <div className="space-y-4 py-4">