From a98d2e21c4e573ec2dcabe163a1c5b1bf6f19b4e Mon Sep 17 00:00:00 2001 From: lucia <51058748+lucia-gomez@users.noreply.github.com> Date: Wed, 10 Apr 2024 20:10:47 -0400 Subject: [PATCH 1/5] filter PA view to approved bookings --- .../src/client/routes/admin/components/Bookings.tsx | 11 +++++++++++ .../src/client/routes/pa/PAPage.tsx | 2 +- 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx b/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx index c3c90347..c2a3b421 100644 --- a/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx +++ b/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx @@ -7,6 +7,7 @@ import getBookingStatus from '../hooks/getBookingStatus'; interface BookingsProps { showNnumber: boolean; + isPaView?: boolean; isUserView?: boolean; } @@ -18,13 +19,23 @@ const TableHeader = (text: string) => ( export const Bookings: React.FC = ({ showNnumber = false, + isPaView = false, isUserView = false, }) => { const { bookings, bookingStatuses, userEmail } = useContext(DatabaseContext); const filteredBookings = useMemo(() => { + const paViewStatuses = [ + BookingStatusLabel.APPROVED, + BookingStatusLabel.CHECKED_IN, + BookingStatusLabel.NO_SHOW, + ]; if (isUserView) return bookings.filter((booking) => booking.email === userEmail); + if (isPaView) + return bookings.filter((booking) => + paViewStatuses.includes(getBookingStatus(booking, bookingStatuses)) + ); return bookings; }, [isUserView, bookings]); diff --git a/media_commons_booking_app/src/client/routes/pa/PAPage.tsx b/media_commons_booking_app/src/client/routes/pa/PAPage.tsx index 32c83d07..d47eb072 100644 --- a/media_commons_booking_app/src/client/routes/pa/PAPage.tsx +++ b/media_commons_booking_app/src/client/routes/pa/PAPage.tsx @@ -59,7 +59,7 @@ const PAPage = () => { {tab === 'safety_training' && } - {tab === 'bookings' && } + {tab === 'bookings' && } )} From 9d5f03bb3c377adcad8c15a965dccc1a1489f01c Mon Sep 17 00:00:00 2001 From: lucia <51058748+lucia-gomez@users.noreply.github.com> Date: Wed, 10 Apr 2024 20:14:28 -0400 Subject: [PATCH 2/5] optimistically update sheet state --- .../client/routes/admin/components/Admin.tsx | 2 +- .../admin/components/BookingActions.tsx | 104 ++++++++++++---- .../admin/components/BookingTableRow.tsx | 103 ++++++++++++++++ .../routes/admin/components/Bookings.tsx | 116 ++---------------- .../src/client/routes/components/AddEmail.tsx | 4 +- .../routes/components/EmailListTable.tsx | 60 +++------ .../routes/components/EmailListTableRow.tsx | 88 +++++++++++++ .../routes/myBookings/myBookingsPage.tsx | 2 +- 8 files changed, 297 insertions(+), 182 deletions(-) create mode 100644 media_commons_booking_app/src/client/routes/admin/components/BookingTableRow.tsx create mode 100644 media_commons_booking_app/src/client/routes/components/EmailListTableRow.tsx diff --git a/media_commons_booking_app/src/client/routes/admin/components/Admin.tsx b/media_commons_booking_app/src/client/routes/admin/components/Admin.tsx index cea5f0f5..6be3937e 100644 --- a/media_commons_booking_app/src/client/routes/admin/components/Admin.tsx +++ b/media_commons_booking_app/src/client/routes/admin/components/Admin.tsx @@ -114,7 +114,7 @@ export default function Admin() { {tab === 'adminUsers' && } {tab === 'paUsers' && } {tab === 'liaesons' && } - {tab === 'bookings' && } + {tab === 'bookings' && } )} diff --git a/media_commons_booking_app/src/client/routes/admin/components/BookingActions.tsx b/media_commons_booking_app/src/client/routes/admin/components/BookingActions.tsx index b1e682f0..07872717 100644 --- a/media_commons_booking_app/src/client/routes/admin/components/BookingActions.tsx +++ b/media_commons_booking_app/src/client/routes/admin/components/BookingActions.tsx @@ -8,11 +8,16 @@ import { useLocation } from 'react-router'; interface Props { calendarEventId: string; + setOptimisticStatus: (x: BookingStatusLabel) => void; status: BookingStatusLabel; } -export default function BookingActions({ status, calendarEventId }: Props) { - const [loading, setLoading] = useState(false); +export default function BookingActions({ + status, + calendarEventId, + setOptimisticStatus, +}: Props) { + const [uiLoading, setUiLoading] = useState(false); const { reloadBookings, reloadBookingStatuses } = useContext(DatabaseContext); const location = useLocation(); @@ -22,19 +27,30 @@ export default function BookingActions({ status, calendarEventId }: Props) { await Promise.all([reloadBookings(), reloadBookingStatuses()]); }; - const ActionButton = (text: string, action: () => Promise) => ( + const onError = () => alert(); + + const ActionButton = ( + text: string, + action: () => Promise, + optimisticNextStatus: BookingStatusLabel + ) => ( ); - if (loading) { + if (uiLoading) { return ( @@ -50,39 +66,73 @@ export default function BookingActions({ status, calendarEventId }: Props) { ); } - const paBtns = ( - <> - {status !== BookingStatusLabel.CHECKED_IN && - ActionButton('Check In', () => - serverFunctions.checkin(calendarEventId) - )} - {status !== BookingStatusLabel.NO_SHOW && - ActionButton('No Show', () => serverFunctions.noShow(calendarEventId))} - - ); + const paBtns = () => { + const checkInBtn = ActionButton( + 'Check In', + () => serverFunctions.checkin(calendarEventId), + BookingStatusLabel.CHECKED_IN + ); + const noShowBtn = ActionButton( + 'No Show', + () => serverFunctions.noShow(calendarEventId), + BookingStatusLabel.NO_SHOW + ); + + if (status === BookingStatusLabel.APPROVED) { + return ( + <> + {checkInBtn} + {noShowBtn} + + ); + } else if (status === BookingStatusLabel.CHECKED_IN) { + return noShowBtn; + } else if (status === BookingStatusLabel.NO_SHOW) { + return checkInBtn; + } + }; if (!isAdminPage) { return ( -
{paBtns}
+
{paBtns()}
); } + if ( + status === BookingStatusLabel.CANCELED || + status === BookingStatusLabel.REJECTED + ) { + return

; + } + return ( - +
{status === BookingStatusLabel.PRE_APPROVED && - ActionButton('2nd Approve', () => - serverFunctions.approveBooking(calendarEventId) + ActionButton( + '2nd Approve', + () => serverFunctions.approveBooking(calendarEventId), + BookingStatusLabel.APPROVED )} {status === BookingStatusLabel.REQUESTED && - ActionButton('1st Approve', () => - serverFunctions.approveBooking(calendarEventId) + ActionButton( + '1st Approve', + () => serverFunctions.approveBooking(calendarEventId), + BookingStatusLabel.PRE_APPROVED )} - {ActionButton('Reject', () => serverFunctions.reject(calendarEventId))} - {ActionButton('Cancel', () => serverFunctions.cancel(calendarEventId))} - {paBtns} + {ActionButton( + 'Reject', + () => serverFunctions.reject(calendarEventId), + BookingStatusLabel.REJECTED + )} + {ActionButton( + 'Cancel', + () => serverFunctions.cancel(calendarEventId), + BookingStatusLabel.CANCELED + )} + {paBtns()}
); diff --git a/media_commons_booking_app/src/client/routes/admin/components/BookingTableRow.tsx b/media_commons_booking_app/src/client/routes/admin/components/BookingTableRow.tsx new file mode 100644 index 00000000..056867f9 --- /dev/null +++ b/media_commons_booking_app/src/client/routes/admin/components/BookingTableRow.tsx @@ -0,0 +1,103 @@ +import { Booking, BookingStatusLabel } from '../../../../types'; +import React, { useContext, useState } from 'react'; + +import BookingActions from './BookingActions'; +import { DatabaseContext } from '../../components/Provider'; +import { formatDate } from '../../../utils/date'; +import getBookingStatus from '../hooks/getBookingStatus'; + +interface Props { + booking: Booking; + isAdminView: boolean; + isUserView: boolean; +} + +export default function BookingTableRow({ + booking, + isAdminView, + isUserView, +}: Props) { + const { bookingStatuses } = useContext(DatabaseContext); + const status = getBookingStatus(booking, bookingStatuses); + + const [optimisticStatus, setOptimisticStatus] = + useState(); + + return ( + + {!isUserView && ( + + )} + {optimisticStatus ?? status} + {booking.roomId} + +
+
+
+ {booking.firstName} {booking.lastName} +
+
{booking.email}
+
+ {booking.phoneNumber} +
+
+
+ + +
+
{formatDate(booking.startDate)}
+
+ + +
+
{formatDate(booking.endDate)}
+
+ + {booking.secondaryName} + {isAdminView && {booking.nNumber}} + {booking.netId} + {booking.department} + {booking.role} + + {booking.sponsorFirstName} {booking.sponsorLastName} + + {booking.sponsorEmail} + {booking.title} + {booking.description} + {booking.expectedAttendance} + {booking.attendeeAffiliation} + + {booking.roomSetup} + {booking.setupDetails && ( + <> +
+ Details +
+ {booking.setupDetails} + + )} + + {booking.chartFieldForRoomSetup} + + {booking.mediaServices} + {booking.mediaServicesDetails && ( + <> +
+ Details +
+ {booking.mediaServicesDetails} + + )} + + {booking.catering} + {booking.cateringService} + {booking.chartFieldForCatering} + {booking.hireSecurity} + {booking.chartFieldForSecurity} + + ); +} diff --git a/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx b/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx index c2a3b421..81b684e1 100644 --- a/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx +++ b/media_commons_booking_app/src/client/routes/admin/components/Bookings.tsx @@ -1,12 +1,12 @@ import React, { useContext, useMemo } from 'react'; -import BookingActions from './BookingActions'; +import { BookingStatusLabel } from '../../../../types'; +import BookingTableRow from './BookingTableRow'; import { DatabaseContext } from '../../components/Provider'; -import { formatDate } from '../../../utils/date'; import getBookingStatus from '../hooks/getBookingStatus'; interface BookingsProps { - showNnumber: boolean; + isAdminView?: boolean; isPaView?: boolean; isUserView?: boolean; } @@ -18,7 +18,7 @@ const TableHeader = (text: string) => ( ); export const Bookings: React.FC = ({ - showNnumber = false, + isAdminView = false, isPaView = false, isUserView = false, }) => { @@ -52,9 +52,8 @@ export const Bookings: React.FC = ({ {TableHeader('Booking Start')} {TableHeader('Booking End')} {TableHeader('Secondary Name')} - {showNnumber && TableHeader('N Number')} + {isAdminView && TableHeader('N Number')} {TableHeader('Net ID')} - {/* {TableHeader('Phone Number')} */} {TableHeader('Department')} {TableHeader('Role')} {TableHeader('Sponsor Name')} @@ -74,105 +73,12 @@ export const Bookings: React.FC = ({ - {filteredBookings.map((booking, index) => { - const status = getBookingStatus(booking, bookingStatuses); - return ( - - {!isUserView && ( - - )} - {status} - {booking.roomId} - -
-
-
- {booking.firstName} {booking.lastName} -
-
- {booking.email} -
-
- {booking.phoneNumber} -
-
-
- - -
-
{formatDate(booking.startDate)}
-
- - -
-
{formatDate(booking.endDate)}
-
- - {booking.secondaryName} - {showNnumber && ( - {booking.nNumber} - )} - {booking.netId} - {/* {booking.phoneNumber} */} - {booking.department} - {booking.role} - - {booking.sponsorFirstName} {booking.sponsorLastName} - - {booking.sponsorEmail} - {booking.title} - - {booking.description} - - - {booking.expectedAttendance} - - - {booking.attendeeAffiliation} - - - {booking.roomSetup} - {booking.setupDetails && ( - <> -
- Details -
- {booking.setupDetails} - - )} - - - {booking.chartFieldForRoomSetup} - - - {booking.mediaServices} - {booking.mediaServicesDetails && ( - <> -
- Details -
- {booking.mediaServicesDetails} - - )} - - {booking.catering} - {booking.cateringService} - - {booking.chartFieldForCatering} - - {booking.hireSecurity} - - {booking.chartFieldForSecurity} - - - ); - })} + {filteredBookings.map((booking, index) => ( + + ))} diff --git a/media_commons_booking_app/src/client/routes/components/AddEmail.tsx b/media_commons_booking_app/src/client/routes/components/AddEmail.tsx index c8e8697e..03e08f99 100644 --- a/media_commons_booking_app/src/client/routes/components/AddEmail.tsx +++ b/media_commons_booking_app/src/client/routes/components/AddEmail.tsx @@ -19,7 +19,7 @@ export default function AddEmail({ userList, userListRefresh, }: Props) { - const [emailToAdd, setEmailToAdd] = useState(); + const [emailToAdd, setEmailToAdd] = useState(''); const [loading, setLoading] = useState(false); const userEmails = useMemo( @@ -28,7 +28,7 @@ export default function AddEmail({ ); const addUser = async () => { - if (!emailToAdd) return; + if (!emailToAdd || emailToAdd.length === 0) return; if (userEmails.includes(emailToAdd)) { alert('This user is already registered'); diff --git a/media_commons_booking_app/src/client/routes/components/EmailListTable.tsx b/media_commons_booking_app/src/client/routes/components/EmailListTable.tsx index 12e5d5a2..8d8fa440 100644 --- a/media_commons_booking_app/src/client/routes/components/EmailListTable.tsx +++ b/media_commons_booking_app/src/client/routes/components/EmailListTable.tsx @@ -1,6 +1,6 @@ -import React, { useMemo, useState } from 'react'; +import React, { useMemo } from 'react'; -import Loading from '../../utils/Loading'; +import EmailListTableRow from './EmailListTableRow'; import { TableNames } from '../../../policy'; // This is a wrapper for google.script.run that lets us use promises. import { serverFunctions } from '../../utils/serverFunctions'; @@ -20,8 +20,6 @@ export default function EmailListTable(props: Props) { const refresh = props.userListRefresh; const columnFormatters = props.columnFormatters || {}; - const [loading, setLoading] = useState(false); - const columnNames = useMemo(() => { if (props.userList.length === 0) { return []; @@ -29,23 +27,6 @@ export default function EmailListTable(props: Props) { return Object.keys(props.userList[0]) as Array as string[]; }, [props.userList]); - const onRemove = async (user: T) => { - setLoading(true); - try { - await serverFunctions.removeFromListByEmail(props.tableName, user.email); - await refresh(); - } catch (ex) { - console.error(ex); - alert('Failed to remove user'); - } finally { - setLoading(false); - } - }; - - if (loading) { - return ; - } - if (props.userList.length === 0) { return

No results

; } @@ -68,31 +49,18 @@ export default function EmailListTable(props: Props) { - {props.userList.map((user, index: number) => { - return ( - - {/* all column values */} - {columnNames.map((columnName, idx) => ( - - {columnFormatters[columnName] - ? columnFormatters[columnName](user[columnName]) - : user[columnName]} - - ))} - - - - - ); - })} + {props.userList.map((user, index: number) => ( + + serverFunctions.removeFromListByEmail( + props.tableName, + user.email + ) + } + {...{ columnNames, columnFormatters, index, user, refresh }} + /> + ))} diff --git a/media_commons_booking_app/src/client/routes/components/EmailListTableRow.tsx b/media_commons_booking_app/src/client/routes/components/EmailListTableRow.tsx new file mode 100644 index 00000000..f2368ec1 --- /dev/null +++ b/media_commons_booking_app/src/client/routes/components/EmailListTableRow.tsx @@ -0,0 +1,88 @@ +import React, { useMemo, useState } from 'react'; + +import Loading from '../../utils/Loading'; +import { TableNames } from '../../../policy'; +// This is a wrapper for google.script.run that lets us use promises. +import { serverFunctions } from '../../utils/serverFunctions'; + +interface EmailField { + email: string; +} + +interface Props { + columnFormatters?: { [key: string]: (value: string) => string }; + columnNames: string[]; + index: number; + refresh: () => Promise; + removeUser: () => Promise; + user: T; +} + +export default function EmailListTableRow( + props: Props +) { + const { columnFormatters, columnNames, index, refresh, removeUser, user } = + props; + const [uiLoading, setUiLoading] = useState(false); + const [isRemoved, setIsRemoved] = useState(false); + + const onError = () => { + alert('Failed to remove user: ' + user.email); + }; + + /** + * Google Sheets API writes are slow. + * To avoid UI lag, assume write will succeed and optimistically remove UI element before response completes. + * If write fails, alert the user and restore UI. + * Only devs should see this error behavior unless something is very broken + */ + const onRemove = async () => { + setUiLoading(true); + setIsRemoved(true); // optimistically hide component + try { + removeUser() + .catch(() => { + onError(); + setIsRemoved(false); + }) + .finally(refresh); + } catch (ex) { + console.error(ex); + onError(); + } finally { + setUiLoading(false); + } + }; + + if (uiLoading) { + return ; + } + + if (isRemoved) { + return null; + } + + return ( + + {/* all column values */} + {columnNames.map((columnName, idx) => ( + + {columnFormatters[columnName] + ? columnFormatters[columnName](user[columnName]) + : user[columnName]} + + ))} + + + + + ); +} diff --git a/media_commons_booking_app/src/client/routes/myBookings/myBookingsPage.tsx b/media_commons_booking_app/src/client/routes/myBookings/myBookingsPage.tsx index 5c2d7962..d1e1346d 100644 --- a/media_commons_booking_app/src/client/routes/myBookings/myBookingsPage.tsx +++ b/media_commons_booking_app/src/client/routes/myBookings/myBookingsPage.tsx @@ -2,5 +2,5 @@ import { Bookings } from '../admin/components/Bookings'; import React from 'react'; export default function MyBookingsPage() { - return ; + return ; } From 6a707572949a066c90a5e42fadc497e561473cfd Mon Sep 17 00:00:00 2001 From: "riho.takagi" Date: Thu, 11 Apr 2024 12:19:13 -0400 Subject: [PATCH 3/5] Update logic about media services --- .../routes/booking/components/FormInput.tsx | 23 +++++++++---------- 1 file changed, 11 insertions(+), 12 deletions(-) diff --git a/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx b/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx index 47febc80..5f332d43 100644 --- a/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx +++ b/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx @@ -568,18 +568,17 @@ const FormInput = ({ handleParentSubmit }) => { (For 220-224) Using DMX lights in ceiling grid )} - {roomNumber.includes('202') || - (roomNumber.includes('1201') && ( - - ))} + {roomNumber.some((room) => [202, 1201].includes(Number(room))) && ( + + )} {watch('mediaServices') !== undefined && From a898d3c173cce2985b94cec06bd6754a55163b21 Mon Sep 17 00:00:00 2001 From: "riho.takagi" Date: Thu, 11 Apr 2024 12:23:30 -0400 Subject: [PATCH 4/5] Update room numbers for checking equipment --- .../routes/booking/components/FormInput.tsx | 24 +++++++++++-------- 1 file changed, 14 insertions(+), 10 deletions(-) diff --git a/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx b/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx index 5f332d43..a6aa8892 100644 --- a/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx +++ b/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx @@ -518,14 +518,18 @@ const FormInput = ({ handleParentSubmit }) => { )}
- + {roomNumber.some((room) => + [103, 220, 221, 222, 223, 224, 230, 233, 260].includes(Number(room)) + ) && ( + + )} {roomNumber.includes('103') && ( )}
From d0f33e817aabed948d0f3554e3b9abdc2d96bf99 Mon Sep 17 00:00:00 2001 From: "riho.takagi" Date: Thu, 11 Apr 2024 12:36:50 -0400 Subject: [PATCH 5/5] Set reply email address --- media_commons_booking_app/src/server/emails.ts | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/media_commons_booking_app/src/server/emails.ts b/media_commons_booking_app/src/server/emails.ts index af8169b5..31d37d4f 100644 --- a/media_commons_booking_app/src/server/emails.ts +++ b/media_commons_booking_app/src/server/emails.ts @@ -7,7 +7,10 @@ export const sendTextEmail = ( body: string ) => { const subj = `${status}: Media Commons request for \"${eventTitle}\"`; - GmailApp.sendEmail(targetEmail, subj, body); + const options = { + replyTo: 'mediacommons.reservations@nyu.edu', + }; + GmailApp.sendEmail(targetEmail, subj, body, options); }; const getEmailBranchTag = () => { @@ -29,7 +32,6 @@ export const sendHTMLEmail = ( eventTitle: string, body ) => { - console.log('contents', contents); const subj = `${getEmailBranchTag()}${status}: Media Commons request for \"${eventTitle}\"`; const htmlTemplate = HtmlService.createTemplateFromFile(templateName); for (const key in contents) { @@ -38,6 +40,7 @@ export const sendHTMLEmail = ( const htmlBody = htmlTemplate.evaluate().getContent(); const options = { htmlBody, + replyTo: 'mediacommons.reservations@nyu.edu', }; GmailApp.sendEmail(targetEmail, subj, body, options); };