summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMichael Hunteman <michael@huntm.net>2024-03-09 15:13:45 -0800
committerMichael Hunteman <michael@huntm.net>2024-03-09 15:13:45 -0800
commit5e5560b558ff818546c94135bebf334be770c920 (patch)
tree23098a45caa1c9153a3387721151d1901aebf390
parenta9333dc90f56ae4e19fabff4822ac1ffba7c6205 (diff)
Add carousel
-rw-r--r--.editorconfig14
-rw-r--r--.prettierrc5
-rw-r--r--index.html4
-rw-r--r--package-lock.json15
-rw-r--r--package.json1
-rw-r--r--src/ThemeContextProvider.tsx12
-rw-r--r--src/apiSlice.ts51
-rw-r--r--src/components/Admin.tsx6
-rw-r--r--src/components/Desktop.tsx6
-rw-r--r--src/components/Home.tsx76
-rw-r--r--src/components/Mobile.tsx17
-rw-r--r--src/components/RsvpForm.tsx63
-rw-r--r--src/components/active.css6
-rw-r--r--src/features/auth/GuestLogin.tsx50
-rw-r--r--src/features/auth/authSlice.ts10
-rw-r--r--src/main.tsx28
-rw-r--r--src/mocks/handlers.ts44
-rw-r--r--src/pages.ts4
-rw-r--r--src/store.ts10
19 files changed, 271 insertions, 151 deletions
diff --git a/.editorconfig b/.editorconfig
index 135ea6f..41c2aaf 100644
--- a/.editorconfig
+++ b/.editorconfig
@@ -1,15 +1,9 @@
root = true
-[*.{ts,tsx,json}]
+[*]
+indent_size = 2
+indent_style = space
end_of_line = lf
-insert_final_newline = true
charset = utf-8
trim_trailing_whitespace = true
-indent_style = space
-indent_size = 2
-# max_line_length = 80
-
-[*.md]
-indent_style = space
-indent_size = 2
-trim_trailing_whitespace = false
+insert_final_newline = true \ No newline at end of file
diff --git a/.prettierrc b/.prettierrc
new file mode 100644
index 0000000..e537c8a
--- /dev/null
+++ b/.prettierrc
@@ -0,0 +1,5 @@
+{
+ "singleQuote": true,
+ "trailingComma": "es5",
+ "arrowParens": "always"
+}
diff --git a/index.html b/index.html
index dfcfa2d..a24cf1f 100644
--- a/index.html
+++ b/index.html
@@ -1,8 +1,8 @@
-<!doctype html>
+<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
- <link rel="icon" type="image/svg+xml" href="/avatar.png" />
+ <link rel="icon" href="/avatar.png" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Wedding</title>
</head>
diff --git a/package-lock.json b/package-lock.json
index 35bab87..9487398 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -15,6 +15,7 @@
"@mui/lab": "^5.0.0-alpha.165",
"@mui/material": "^5.15.9",
"@reduxjs/toolkit": "^2.2.1",
+ "prettier": "^3.2.5",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-hook-form": "^7.50.1",
@@ -4065,6 +4066,20 @@
"node": ">= 0.8.0"
}
},
+ "node_modules/prettier": {
+ "version": "3.2.5",
+ "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.2.5.tgz",
+ "integrity": "sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A==",
+ "bin": {
+ "prettier": "bin/prettier.cjs"
+ },
+ "engines": {
+ "node": ">=14"
+ },
+ "funding": {
+ "url": "https://github.com/prettier/prettier?sponsor=1"
+ }
+ },
"node_modules/prop-types": {
"version": "15.8.1",
"resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz",
diff --git a/package.json b/package.json
index 0156564..3edb6db 100644
--- a/package.json
+++ b/package.json
@@ -17,6 +17,7 @@
"@mui/lab": "^5.0.0-alpha.165",
"@mui/material": "^5.15.9",
"@reduxjs/toolkit": "^2.2.1",
+ "prettier": "^3.2.5",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-hook-form": "^7.50.1",
diff --git a/src/ThemeContextProvider.tsx b/src/ThemeContextProvider.tsx
index 63f4e81..dc3cb89 100644
--- a/src/ThemeContextProvider.tsx
+++ b/src/ThemeContextProvider.tsx
@@ -4,14 +4,14 @@ import useMediaQuery from '@mui/material/useMediaQuery';
type ThemeContextType = {
toggleColorMode: () => void;
-}
+};
type ThemeProviderProps = {
children: ReactNode;
-}
+};
export const ThemeContext = createContext<ThemeContextType>({
- toggleColorMode: () => {}
+ toggleColorMode: () => {},
});
function ThemeContextProvider({ children }: ThemeProviderProps) {
@@ -29,16 +29,14 @@ function ThemeContextProvider({ children }: ThemeProviderProps) {
createTheme({
palette: {
mode,
- }
+ },
}),
[mode]
);
return (
<ThemeContext.Provider value={{ toggleColorMode }}>
- <ThemeProvider theme={theme}>
- {children}
- </ThemeProvider>
+ <ThemeProvider theme={theme}>{children}</ThemeProvider>
</ThemeContext.Provider>
);
}
diff --git a/src/apiSlice.ts b/src/apiSlice.ts
index 6a1196b..f337fdd 100644
--- a/src/apiSlice.ts
+++ b/src/apiSlice.ts
@@ -1,23 +1,23 @@
import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react';
-import { RootState } from './store'
+import { RootState } from './store';
export interface LoginRequest {
- firstName: string
- lastName: string
+ firstName: string;
+ lastName: string;
}
export interface LoginResponse {
- guest: Guest
- token: string
+ guest: Guest;
+ token: string;
}
export interface Guest {
- id: number
- firstName: string
- lastName: string
- attendance: string
- email: string
- message: string
+ id: number;
+ firstName: string;
+ lastName: string;
+ attendance: string;
+ email: string;
+ message: string;
}
export const apiSlice = createApi({
@@ -30,34 +30,31 @@ export const apiSlice = createApi({
headers.set('authorization', `Bearer ${token}`);
}
return headers;
- }
+ },
}),
tagTypes: ['Guests'],
- endpoints: builder => ({
+ endpoints: (builder) => ({
getGuests: builder.query<void, void>({
query: () => '/guests',
- providesTags: ['Guests']
+ providesTags: ['Guests'],
}),
updateGuest: builder.mutation<Guest, Guest>({
- query: guest => ({
+ query: (guest) => ({
url: `/guests/${guest?.id}`,
method: 'PATCH',
body: guest,
- providesTags: ['Guests']
- })
+ providesTags: ['Guests'],
+ }),
}),
login: builder.mutation<LoginResponse, LoginRequest>({
- query: credentials => ({
+ query: (credentials) => ({
url: '/guest-login',
method: 'POST',
- body: credentials
- })
- })
- })
+ body: credentials,
+ }),
+ }),
+ }),
});
-export const {
- useGetGuestsQuery,
- useUpdateGuestMutation,
- useLoginMutation
-} = apiSlice;
+export const { useGetGuestsQuery, useUpdateGuestMutation, useLoginMutation } =
+ apiSlice;
diff --git a/src/components/Admin.tsx b/src/components/Admin.tsx
index bd1545d..a3da3fa 100644
--- a/src/components/Admin.tsx
+++ b/src/components/Admin.tsx
@@ -6,17 +6,17 @@ function Admin() {
isLoading,
isSuccess,
isError,
- error
+ error,
} = useGetGuestsQuery();
let content;
if (isLoading) {
- content = <p>Loading...</p>
+ content = <p>Loading...</p>;
} else if (isSuccess) {
content = JSON.stringify(guests);
} else if (isError) {
- content = <>{error.toString()}</>
+ content = <>{error.toString()}</>;
}
return (
diff --git a/src/components/Desktop.tsx b/src/components/Desktop.tsx
index 34a0621..983c929 100644
--- a/src/components/Desktop.tsx
+++ b/src/components/Desktop.tsx
@@ -1,6 +1,6 @@
import { useContext } from 'react';
-import { Link } from "react-router-dom";
-import { Button, IconButton } from "@mui/material";
+import { Link } from 'react-router-dom';
+import { Button, IconButton } from '@mui/material';
import DarkModeIcon from '@mui/icons-material/DarkMode';
import LightModeIcon from '@mui/icons-material/LightMode';
import { useTheme } from '@mui/material/styles';
@@ -13,7 +13,7 @@ function Desktop() {
return (
<div style={{ marginLeft: 'auto' }}>
- {pages.map(page => (
+ {pages.map((page) => (
<Button color="inherit" component={Link} to={page?.to} key={page?.name}>
{page?.name}
</Button>
diff --git a/src/components/Home.tsx b/src/components/Home.tsx
new file mode 100644
index 0000000..da9eb6b
--- /dev/null
+++ b/src/components/Home.tsx
@@ -0,0 +1,76 @@
+import { useEffect, useRef, useState } from 'react';
+import './active.css';
+
+function Home() {
+ const [index, setIndex] = useState(0);
+ const colors = ['#FF0000', '#00FF00', '#0000FF'];
+ const timeout = useRef(0);
+
+ useEffect(() => {
+ resetTimeout();
+ timeout.current = window.setTimeout(
+ () =>
+ setIndex((prevIndex) =>
+ prevIndex === colors.length - 1 ? 0 : prevIndex + 1
+ ),
+ 2500
+ );
+
+ return () => {
+ resetTimeout();
+ };
+ }, [index]);
+
+ const resetTimeout = () => {
+ if (timeout.current) {
+ clearTimeout(timeout.current);
+ }
+ };
+
+ return (
+ <>
+ <div style={{ margin: 'auto', overflow: 'hidden' }}>
+ <div
+ style={{
+ whiteSpace: 'nowrap',
+ transform: `translate3d(${-index * 100}%, 0, 0)`,
+ transition: 'ease 1000ms',
+ }}
+ >
+ {colors.map((backgroundColor, colorIndex) => (
+ <div
+ key={colorIndex}
+ style={{
+ display: 'inline-block',
+ backgroundColor,
+ height: '80vh',
+ width: '100%',
+ }}
+ />
+ ))}
+ </div>
+ <div style={{ textAlign: 'center' }}>
+ {colors.map((_, colorIndex) => (
+ <div
+ key={colorIndex}
+ style={{
+ display: 'inline-block',
+ height: '0.8rem',
+ width: '0.8rem',
+ borderRadius: '50%',
+ cursor: 'pointer',
+ margin: '0.7rem 0.7rem 0.7rem',
+ }}
+ className={colorIndex === index ? 'active' : 'inactive'}
+ onClick={() => {
+ setIndex(colorIndex);
+ }}
+ />
+ ))}
+ </div>
+ </div>
+ </>
+ );
+}
+
+export default Home;
diff --git a/src/components/Mobile.tsx b/src/components/Mobile.tsx
index f2ca255..f5502c0 100644
--- a/src/components/Mobile.tsx
+++ b/src/components/Mobile.tsx
@@ -1,7 +1,7 @@
import { useContext } from 'react';
-import { Link } from "react-router-dom";
+import { Link } from 'react-router-dom';
import { useState } from 'react';
-import { Button, IconButton, Menu, MenuItem } from "@mui/material";
+import { Button, IconButton, Menu, MenuItem } from '@mui/material';
import DarkModeIcon from '@mui/icons-material/DarkMode';
import LightModeIcon from '@mui/icons-material/LightMode';
import { useTheme } from '@mui/material/styles';
@@ -27,13 +27,22 @@ function Mobile() {
<IconButton color="inherit" onClick={toggleColorMode}>
{theme.palette.mode === 'dark' ? <DarkModeIcon /> : <LightModeIcon />}
</IconButton>
- <IconButton color="inherit" sx={{ ml: 'auto' }} onClick={handleOpenNavMenu}>
+ <IconButton
+ color="inherit"
+ sx={{ ml: 'auto' }}
+ onClick={handleOpenNavMenu}
+ >
<MenuIcon />
</IconButton>
<Menu anchorEl={anchorEl} open={!!anchorEl} onClose={handleCloseNavMenu}>
{pages.map((page) => (
<MenuItem key={page.name} onClick={handleCloseNavMenu}>
- <Button color="inherit" component={Link} to={page?.to} key={page?.name}>
+ <Button
+ color="inherit"
+ component={Link}
+ to={page?.to}
+ key={page?.name}
+ >
{page?.name}
</Button>
</MenuItem>
diff --git a/src/components/RsvpForm.tsx b/src/components/RsvpForm.tsx
index d7efecf..090987b 100644
--- a/src/components/RsvpForm.tsx
+++ b/src/components/RsvpForm.tsx
@@ -10,14 +10,19 @@ import {
TextField,
} from '@mui/material';
import { useForm, Controller } from 'react-hook-form';
-import { useOutletContext } from "react-router-dom";
+import { useOutletContext } from 'react-router-dom';
import { useUpdateGuestMutation, Guest } from '../apiSlice';
function RsvpForm() {
const [updateGuest] = useUpdateGuestMutation();
const guest: Guest = useOutletContext();
- const { register, handleSubmit, control, formState: { errors } } = useForm({
+ const {
+ register,
+ handleSubmit,
+ control,
+ formState: { errors },
+ } = useForm({
defaultValues: {
id: guest?.id,
firstName: guest?.firstName,
@@ -25,22 +30,27 @@ function RsvpForm() {
attendance: '',
email: '',
partySize: 1,
- message: ''
- }
+ message: '',
+ },
});
const onSubmit = async (data: Guest) => {
- updateGuest({...data});
+ updateGuest({ ...data });
};
return (
- <Container component="form" maxWidth="sm" noValidate onSubmit={handleSubmit(onSubmit)}>
+ <Container
+ component="form"
+ maxWidth="sm"
+ noValidate
+ onSubmit={handleSubmit(onSubmit)}
+ >
<Grid container spacing={2} sx={{ mt: 8 }}>
<Grid item xs={12}>
<p>
- Please RSVP for the wedding by March 10, 2025.
- The ceremony will commence at 3 pm on April 26 in ...
- The reception will follow at 5 pm in A Venue on the Ridge.
+ Please RSVP for the wedding by March 10, 2025. The ceremony will
+ commence at 3 pm on April 26 in ... The reception will follow at 5
+ pm in A Venue on the Ridge.
</p>
</Grid>
<Grid item xs={12}>
@@ -81,12 +91,13 @@ function RsvpForm() {
error={!!errors.email}
helperText={errors.email?.message}
required
- {...register('email',
- { required: 'This field is required',
- pattern: { value: /\S+@\S+\.\S+/,
- message: 'Please enter a valid email address' }
- }
- )}
+ {...register('email', {
+ required: 'This field is required',
+ pattern: {
+ value: /\S+@\S+\.\S+/,
+ message: 'Please enter a valid email address',
+ },
+ })}
/>
</Grid>
<Grid item xs={12} md={6} lg={6}>
@@ -98,12 +109,14 @@ function RsvpForm() {
error={!!errors.partySize}
helperText={errors.partySize?.message}
required
- {...register('partySize',
- { required: 'This field is required',
- min: { value: 1, message: 'Please enter a positive integer' },
- max: { value: 9, message: 'Please enter an integer less than 10' }
- }
- )}
+ {...register('partySize', {
+ required: 'This field is required',
+ min: { value: 1, message: 'Please enter a positive integer' },
+ max: {
+ value: 9,
+ message: 'Please enter an integer less than 10',
+ },
+ })}
/>
</Grid>
<Grid item xs={12}>
@@ -118,9 +131,11 @@ function RsvpForm() {
</Grid>
<Grid item xs={12}>
<div
- style={{ display: 'flex',
- flexDirection: 'column',
- alignItems: 'center' }}
+ style={{
+ display: 'flex',
+ flexDirection: 'column',
+ alignItems: 'center',
+ }}
>
<Button type="submit" variant="contained">
RSVP
diff --git a/src/components/active.css b/src/components/active.css
new file mode 100644
index 0000000..60feeea
--- /dev/null
+++ b/src/components/active.css
@@ -0,0 +1,6 @@
+.active {
+ background-color: #1976d2;
+}
+.inactive {
+ background-color: #c4c4c4;
+} \ No newline at end of file
diff --git a/src/features/auth/GuestLogin.tsx b/src/features/auth/GuestLogin.tsx
index 11f3d5b..94bad05 100644
--- a/src/features/auth/GuestLogin.tsx
+++ b/src/features/auth/GuestLogin.tsx
@@ -10,33 +10,42 @@ function GuestLogin() {
const navigate = useNavigate();
const [login] = useLoginMutation();
- const { register, handleSubmit, formState: { errors } } = useForm<LoginRequest>({
+ const {
+ register,
+ handleSubmit,
+ formState: { errors },
+ } = useForm<LoginRequest>({
defaultValues: {
firstName: '',
- lastName: ''
- }
+ lastName: '',
+ },
});
const onSubmit = async (data: LoginRequest) => {
- try {
- dispatch(setCredentials(await login(data).unwrap()));
- navigate('/rsvp');
- } catch (e) {
- console.log(e);
- }
+ try {
+ dispatch(setCredentials(await login(data).unwrap()));
+ navigate('/rsvp');
+ } catch (e) {
+ console.log(e);
+ }
};
return (
- <Container component="form" maxWidth="xs" noValidate onSubmit={handleSubmit(onSubmit)}>
+ <Container
+ component="form"
+ maxWidth="xs"
+ noValidate
+ onSubmit={handleSubmit(onSubmit)}
+ >
<div
- style={{ marginTop: 80,
- display: 'flex',
- flexDirection: 'column',
- alignItems: 'center' }}
+ style={{
+ marginTop: 80,
+ display: 'flex',
+ flexDirection: 'column',
+ alignItems: 'center',
+ }}
>
- <Typography variant="h6">
- Guest Login
- </Typography>
+ <Typography variant="h6">Guest Login</Typography>
<TextField
label="First Name"
variant="outlined"
@@ -57,12 +66,7 @@ function GuestLogin() {
required
{...register('lastName', { required: 'This field is required' })}
/>
- <Button
- type="submit"
- variant="contained"
- fullWidth
- sx={{ mt: 2 }}
- >
+ <Button type="submit" variant="contained" fullWidth sx={{ mt: 2 }}>
Log in
</Button>
</div>
diff --git a/src/features/auth/authSlice.ts b/src/features/auth/authSlice.ts
index d5b294c..34ede58 100644
--- a/src/features/auth/authSlice.ts
+++ b/src/features/auth/authSlice.ts
@@ -3,9 +3,9 @@ import type { RootState } from '../../store';
import { Guest } from '../../apiSlice';
type AuthState = {
- guest: Guest | null
- token: string | null
-}
+ guest: Guest | null;
+ token: string | null;
+};
const authSlice = createSlice({
name: 'auth',
@@ -15,8 +15,8 @@ const authSlice = createSlice({
const { guest, token } = action.payload;
state.guest = guest;
state.token = token;
- }
- }
+ },
+ },
});
export const { setCredentials } = authSlice.actions;
diff --git a/src/main.tsx b/src/main.tsx
index 3125ede..8aba432 100644
--- a/src/main.tsx
+++ b/src/main.tsx
@@ -4,35 +4,39 @@ import { createBrowserRouter, RouterProvider } from 'react-router-dom';
import { Provider } from 'react-redux';
import App from './App';
import store from './store';
-import ThemeContextProvider from './ThemeContextProvider'
+import ThemeContextProvider from './ThemeContextProvider';
import Schedule from './components/Schedule';
import Registry from './components/Registry';
import GuestLogin from './features/auth/GuestLogin';
import Rsvp from './components/Rsvp';
import RsvpForm from './components/RsvpForm';
import Admin from './components/Admin';
+import Home from './components/Home';
const router = createBrowserRouter([
{
- path: '/',
element: <App />,
children: [
{
+ path: '/',
+ element: <Home />,
+ },
+ {
path: 'schedule',
- element: <Schedule />
+ element: <Schedule />,
},
{
path: 'registry',
- element: <Registry />
+ element: <Registry />,
},
{
path: 'guest-login',
- element: <GuestLogin />
+ element: <GuestLogin />,
},
{
path: 'admin',
- element: <Admin />
- }
+ element: <Admin />,
+ },
],
},
{
@@ -40,11 +44,11 @@ const router = createBrowserRouter([
children: [
{
path: 'rsvp',
- element: <RsvpForm />
- }
- ]
- }
-], { basename: '/wedding' });
+ element: <RsvpForm />,
+ },
+ ],
+ },
+]);
const enableMocking = async () => {
const { worker } = await import('./mocks/browser');
diff --git a/src/mocks/handlers.ts b/src/mocks/handlers.ts
index 6bd17fa..e3569df 100644
--- a/src/mocks/handlers.ts
+++ b/src/mocks/handlers.ts
@@ -5,32 +5,28 @@ const token = nanoid();
export const handlers = [
http.post('/guest-login', () => {
- return HttpResponse.json(
- {
- guest: {
- id: 1,
- firstName: 'Michael',
- lastName: 'Hunteman',
- attendance: 'false',
- meal: '',
- email: '',
- message: ''
- },
- token,
- }
- )
- }),
- http.patch('/guests/1', () => {
- return HttpResponse.json(
- {
+ return HttpResponse.json({
+ guest: {
id: 1,
firstName: 'Michael',
lastName: 'Hunteman',
- attendance: 'true',
- meal: 'beef',
+ attendance: 'false',
+ meal: '',
email: '',
- message: ''
- }
- )
- })
+ message: '',
+ },
+ token,
+ });
+ }),
+ http.patch('/guests/1', () => {
+ return HttpResponse.json({
+ id: 1,
+ firstName: 'Michael',
+ lastName: 'Hunteman',
+ attendance: 'true',
+ meal: 'beef',
+ email: '',
+ message: '',
+ });
+ }),
];
diff --git a/src/pages.ts b/src/pages.ts
index ce1d29e..5b5d119 100644
--- a/src/pages.ts
+++ b/src/pages.ts
@@ -1,8 +1,8 @@
const pages = [
- { name: 'Schedule', to: '/schedule'},
+ { name: 'Schedule', to: '/schedule' },
{ name: 'RSVP', to: '/guest-login' },
{ name: 'Registry', to: '/registry' },
- { name: 'Admin', to: '/admin' }
+ { name: 'Admin', to: '/admin' },
];
export default pages;
diff --git a/src/store.ts b/src/store.ts
index d3e02d2..264639e 100644
--- a/src/store.ts
+++ b/src/store.ts
@@ -5,12 +5,12 @@ import authReducer from './features/auth/authSlice';
const store = configureStore({
reducer: {
[apiSlice.reducerPath]: apiSlice.reducer,
- auth: authReducer
+ auth: authReducer,
},
- middleware: getDefaultMiddleware =>
- getDefaultMiddleware().concat(apiSlice.middleware)
+ middleware: (getDefaultMiddleware) =>
+ getDefaultMiddleware().concat(apiSlice.middleware),
});
export default store;
-export type RootState = ReturnType<typeof store.getState>
-export type AppDispatch = typeof store.dispatch
+export type RootState = ReturnType<typeof store.getState>;
+export type AppDispatch = typeof store.dispatch;