Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Style and Create User Pop up Modal in Registration Page #33 #34

Merged
merged 10 commits into from
Aug 18, 2024
191 changes: 182 additions & 9 deletions src/components/RegistrationTable/EditCell.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React from 'react'
import React, { useEffect, useState } from 'react'
import { Button } from "@/components/ui/button"
import {
Dialog,
Expand All @@ -11,13 +11,42 @@ import { Row, Table } from "@tanstack/react-table"
import { Attendee } from "./columns"
import Image from "next/image";
import PopoutIcon from "../../../public/assets/icons/popout_icon.svg";
import UserInfo from "./editCellPopUp.tsx/userInfo"
import UserResponses from './editCellPopUp.tsx/userResponses'
import { RegistrationQuestion } from '@/types'
import router from 'next/router'

interface EditCellProps {
row: Row<Attendee>
table: Table<Attendee>
}

export const EditCell: React.FC<EditCellProps> = ({ row }) => {
export const EditCell: React.FC<EditCellProps> = ({ row, table }) => {
const [questions, setQuestions] = useState<RegistrationQuestion[]>([]);

useEffect(() => {
const fetchData = async () => {
if (router.isReady) {
const eventId = router.query.eventId as string;
const attendeeId = row.original.id;
const eventYear = router.query.year as string;

if (eventId && attendeeId && eventYear) {
try {
const data = await fetchQuestionsAndResponses(attendeeId, eventId, eventYear);
setQuestions(data.questions);
} catch (error) {
console.error("Error fetching questions and responses:", error);
}
}
Comment on lines +28 to +41
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This data will already be fetched by src/pages/admin/event/[eventId]/[year]/index.tsx and we can pass it to this Pop-up component instead of needing to re-fetch everything. Leave this in for now though, and we can refactor it once we have the backend working with the Admin Event Registrations page

}
};

fetchData();
}, [row.original.id]);



const handleEdit = () => {
// Handle edit logic here
console.log("Editing row:", row.original)
Expand All @@ -32,18 +61,162 @@ export const EditCell: React.FC<EditCellProps> = ({ row }) => {
</Button>
</DialogTrigger>

<DialogContent className="max-w-md w-full bg-events-active-tab-bg">
<DialogContent className="max-w-[750px] w-full max-h-lg bg-events-active-tab-bg border-0">
<DialogHeader>
<DialogTitle className="text-white">{row.original.firstName} {row.original.lastName}</DialogTitle>
<span className="italic">Form Responses</span>
<span className="italic text-white">Form Responses</span>
</DialogHeader>
{/* divider */}

<div className="w-full h-[1px] bg-[#8DA1D1] my-3"/>

<Button onClick={handleEdit}>Save changes</Button>

<div className="max-w-full h-[1px] bg-divider my-3" />

<div className="max-h-[300px] max-w-full overflow-y-auto">
<UserInfo row={row} table={table} />
{/* divider */}
<div className="max-w-full h-[1px] bg-divider my-3">
<UserResponses questions={questions} responses={row.original.dynamicResponses} />
</div>
</div>
{/* divider */}
<div className="w-full h-[1px] bg-divider my-3" />
{/* Added DialogTrigger here to close dialog upon button click */}
<DialogTrigger asChild>
<Button onClick={handleEdit}>Save changes</Button>
</DialogTrigger>
</DialogContent>
</Dialog>
)
}

async function fetchQuestionsAndResponses (attendeeId: string, eventId: string, eventYear: string) {
// TODO: Replace mock data with actual backend call
const mockQuestions: RegistrationQuestion[] =
[
{
"questionImageUrl": "",
"label": "What school do you attend?",
"questionId": "bede9713-17cf-4bb9-b362-8c30a1e5b543",
"type": "SELECT",
"choices": ["UBC", "SFU", "KPU", "Douglas", "High School", "..."],
"required": true
},
{
"label": "Do you have any accessibility/mobility needs that you would need us to know about?",
"questionId": "49512eca-eec6-4639-b70c-f60f2e0a9371",
"type": "SELECT",
"choices": ["Yes", "No"],
"required": true
},
{
"label": "If you answered 'Yes' to the question above, please let us know of any accommodations you will need below:",
"questionId": "ee2b5b93-3792-4332-ba83-24b995f12094",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "LinkedIn URL:",
"questionId": "013bb98c-4286-4649-bbb9-fbc27185925c",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "Resume:",
"questionId": "ea4a8fa7-7bcf-4f86-af60-27ac61c7680b",
"type": "UPLOAD",
"choices": [],
"required": false
},
{
"label": "Do you consent to us sharing your resume with our attending partners? (If you didn't upload one, select N/A)",
"questionId": "f33ba987-6a5f-4ea2-9ba9-ca5f087e6fef",
"type": "SELECT",
"choices": ["Yes", "No", "N/A"],
"required": true
},
{
"label": "Do you consent to having your email added to our sponsors' mailing lists?",
"questionId": "a04e1064-65b2-4873-bede-9c4d5cbd32c3",
"type": "SELECT",
"choices": ["Yes", "No"],
"required": true
},
{
"label": "Would you be interested in getting your headshot taken at the conference?",
"questionId": "24f0b385-d92e-43be-92a3-2b225964a778",
"type": "SELECT",
"choices": ["Yes", "No"],
"required": true
},
{
"label": "What's the first thing that comes to mind when you hear 'Digital Disruptions'?",
"questionId": "04f58eea-1861-4da3-8e4c-8d8e994ce8ba",
"type": "TEXT",
"choices": [],
"required": true
},
{
"label": "Would you like to participate in our Attendee Showcase (view the event description above for more info)? If so, please complete ALL of the questions from this point on, otherwise, you may skip them.",
"questionId": "cb1a1e83-f581-473c-97e0-48e9a88d7d20",
"type": "SELECT",
"choices": ["I would like to participate", "I would not like to participate"],
"required": true
},
{
"label": "[A/S] Project Name",
"questionId": "16ce18ff-9e88-4988-8965-b3edc172437a",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "[A/S] Full names of those who contributed to this project (you may only submit projects with up to 4 contributors MAX)",
"questionId": "0836c57c-6163-4905-acb5-6578510b5a63",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "[A/S] Brief description of your project",
"charLimit": 350,
"questionId": "5328b778-e593-4077-bd58-d7b0f7cd1752",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "[A/S] Key technologies used to create your project",
"questionId": "10231b92-5503-4a22-a205-25d99bee580b",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "[A/S] Relevant photo of your project (i.e. logo, screenshot of webpage, photo of prototype, etc)",
"questionId": "e44b5fea-b18a-41bf-b6f2-e685b88d74da",
"type": "UPLOAD",
"choices": [],
"required": false
},
{
"label": "[A/S] Link to your project/to learn more about your project (i.e. Github repo, website link, etc)",
"questionId": "1099caf7-6e25-4c9a-97ab-c7628d1507e9",
"type": "TEXT",
"choices": [],
"required": false
},
{
"label": "I confirm that the project I submitted abides by Blueprint's Attendee Showcase requirements (stated in this form's description)",
"questionId": "f83944a8-97e2-421c-903c-aac91106fe34",
"type": "SELECT",
"choices": ["I confirm", "I didn't submit a project"],
"required": true
}
];

return {
questions: mockQuestions
};
};


10 changes: 10 additions & 0 deletions src/components/RegistrationTable/columns.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,16 @@ export const columns: ColumnDef<Attendee>[] = [
accessorKey: "appStatus",
header: "App. Status",
cell: TableCell,
meta: {
type: "select",
options: [ // These values were inferred from the database
{ value: "Accepted", label: "Accepted" },
{ value: "Reviewing", label: "Reviewing" },
{ value: "Waitlist", label: "Waitlist" },
{ value: "Rejected", label: "Rejected" },
],
} as ColumnMeta,
size: 200,
},
{
accessorKey: "firstName",
Expand Down
75 changes: 75 additions & 0 deletions src/components/RegistrationTable/editCellPopUp.tsx/userInfo.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
import React, { ChangeEvent, useEffect, useState } from 'react'
import { Row, Table } from "@tanstack/react-table"
import { Attendee } from "../columns"
import SelectCell from './userPopupEdit'
import { ColumnMeta } from '../columns'

interface EditCellProps {
row: Row<Attendee>,
table: Table<Attendee>
}

const UserInfo: React.FC<EditCellProps> = ({ row, table }) => {
const [fieldLabels, setFieldLabels] = useState<{ [key: string]: string }>({});
const [dropDownList, setDropDownList] = useState<{ [key: string]: string[] }>({});

useEffect(() => {
const generateFieldLabels = () => {
const columns = table.getAllColumns();
const labels: { [key: string]: string } = {};

columns.forEach((column) => {
const accessorKey = column.id;
const header = column.columnDef.header;

if (accessorKey && typeof header === 'string') {
labels[accessorKey] = header;
}
});
return labels;
};

setFieldLabels(generateFieldLabels());

const generateDropDownList = () => {
const columns = table.getAllColumns();
const options: { [key: string]: string[] } = {};
columns.forEach(column => {
// had to import meta as Column Meta or else encountered errors
const meta = column.columnDef.meta as ColumnMeta | undefined;

if (meta?.type === 'select') {
options[column.id] = meta.options?.map(opt => opt.value) || [];
}
});

return options;
};

setDropDownList(generateDropDownList());

}, [table]);

const fieldsToDisplay = Object.keys(row.original).filter(key => key !== 'shouldNotDisplay' && key !== 'id' && key != 'dynamicResponses');


return (
<div className="text-white gap-4 m-3 grid auto-cols-fr sm:grid-cols-2">
{fieldsToDisplay?.map((key) => (
<div key={key}>
<label className="block font-bold text-baby-blue">{fieldLabels[key]}:</label>
{key === 'regStatus' || key === 'appStatus' ? (
<SelectCell originalValue={row.original[key]} dropDownList={dropDownList[key]}/>
) : key === 'points' ? (
<SelectCell originalValue={row.original[key]} dropDownList={dropDownList[key]}/>
) : (
<span>{row.original[key]}</span>
)}
</div>
))}
</div>
)
}

export default UserInfo

Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
import { Select, SelectContent, SelectGroup, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
import { Input } from "@/components/ui/input"
import React, { useState } from 'react'


interface SelectCellProps {
originalValue: string | number,
dropDownList: string[]
}

const SelectCell: React.FC<SelectCellProps> = ({ originalValue, dropDownList }) => {
const [value, setValue] = useState(originalValue)

const onSelectChange = (newValue: string) => {
setValue(newValue)
console.log("TODO - update data")
}

const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setValue(e.target.value);
};


return (
<div>
{dropDownList ? (
<Select onValueChange={onSelectChange} defaultValue="Not Found">
<SelectTrigger className="p3 rounded-none bg-events-active-tab-bg text-white p-0 border-0 border-b-2 border-b-baby-blue">
<SelectValue>
{value}
</SelectValue>
</SelectTrigger>
<SelectContent className='focus:border-0 bg-white'>
<SelectGroup>
{/* Use the key to access the correct dropDownList */}
{dropDownList.map((item) => (
<SelectItem key={item} value={item}>
{item}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
) : (
<Input
className="p3 rounded-none bg-events-active-tab-bg text-white p-0 border-0 border-b-2 border-b-baby-blue"
type="number"
value={value}
onChange={handleInputChange}
/>
)}
</div>
);
}

export default SelectCell;

Loading