,
+ 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 c3c90347..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,13 @@
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;
}
@@ -17,14 +18,24 @@ const TableHeader = (text: string) => (
);
export const Bookings: React.FC = ({
- showNnumber = false,
+ isAdminView = 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]);
@@ -41,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')}
@@ -63,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/booking/components/FormInput.tsx b/media_commons_booking_app/src/client/routes/booking/components/FormInput.tsx
index 47febc80..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') && (
)}
- {roomNumber.includes('202') ||
- (roomNumber.includes('1201') && (
-
- ))}
+ {roomNumber.some((room) => [202, 1201].includes(Number(room))) && (
+
+ )}
{watch('mediaServices') !== undefined &&
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 ;
}
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' && }
)}
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);
};