Commit face2c7e by Madhankumar

seat limit and toaster added

parent 518a7029
{ {
"users": [ "users": [
{ {
"mobile": "12345678",
"id": 1
},
{
"mobile": "123456789",
"isLogged": true,
"id": 2
},
{
"mobile": "1234567891",
"isLogged": true,
"id": 3
},
{
"mobile": "1234567890", "mobile": "1234567890",
"isLogged": true, "id": 1
"id": 4
} }
], ],
"reservedSeats": [ "reservedSeats": [
...@@ -25,7 +10,9 @@ ...@@ -25,7 +10,9 @@
"userId": 1, "userId": 1,
"seats": [ "seats": [
"D1", "D1",
"D4" "D4",
"D2",
"D3"
], ],
"id": 1 "id": 1
} }
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
"react-dom": "^18.2.0", "react-dom": "^18.2.0",
"react-router-dom": "^6.16.0", "react-router-dom": "^6.16.0",
"react-scripts": "5.0.1", "react-scripts": "5.0.1",
"react-tooltip": "^5.21.5", "react-toastify": "^9.1.3",
"web-vitals": "^2.1.4" "web-vitals": "^2.1.4"
}, },
"devDependencies": { "devDependencies": {
...@@ -2945,6 +2945,7 @@ ...@@ -2945,6 +2945,7 @@
"version": "1.5.0", "version": "1.5.0",
"resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.5.0.tgz", "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.5.0.tgz",
"integrity": "sha512-kK1h4m36DQ0UHGj5Ah4db7R0rHemTqqO0QLvUqi1/mUUp3LuAWbWxdxSIf/XsnH9VS6rRVPLJCncjRzUvyCLXg==", "integrity": "sha512-kK1h4m36DQ0UHGj5Ah4db7R0rHemTqqO0QLvUqi1/mUUp3LuAWbWxdxSIf/XsnH9VS6rRVPLJCncjRzUvyCLXg==",
"dev": true,
"dependencies": { "dependencies": {
"@floating-ui/utils": "^0.1.3" "@floating-ui/utils": "^0.1.3"
} }
...@@ -2953,6 +2954,7 @@ ...@@ -2953,6 +2954,7 @@
"version": "1.5.3", "version": "1.5.3",
"resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.5.3.tgz", "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.5.3.tgz",
"integrity": "sha512-ClAbQnEqJAKCJOEbbLo5IUlZHkNszqhuxS4fHAVxRPXPya6Ysf2G8KypnYcOTpx6I8xcgF9bbHb6g/2KpbV8qA==", "integrity": "sha512-ClAbQnEqJAKCJOEbbLo5IUlZHkNszqhuxS4fHAVxRPXPya6Ysf2G8KypnYcOTpx6I8xcgF9bbHb6g/2KpbV8qA==",
"dev": true,
"dependencies": { "dependencies": {
"@floating-ui/core": "^1.4.2", "@floating-ui/core": "^1.4.2",
"@floating-ui/utils": "^0.1.3" "@floating-ui/utils": "^0.1.3"
...@@ -2974,7 +2976,8 @@ ...@@ -2974,7 +2976,8 @@
"node_modules/@floating-ui/utils": { "node_modules/@floating-ui/utils": {
"version": "0.1.4", "version": "0.1.4",
"resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.1.4.tgz", "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.1.4.tgz",
"integrity": "sha512-qprfWkn82Iw821mcKofJ5Pk9wgioHicxcQMxx+5zt5GSKoqdWvgG5AxVmpmUUjzTLPVSH5auBrhI93Deayn/DA==" "integrity": "sha512-qprfWkn82Iw821mcKofJ5Pk9wgioHicxcQMxx+5zt5GSKoqdWvgG5AxVmpmUUjzTLPVSH5auBrhI93Deayn/DA==",
"dev": true
}, },
"node_modules/@humanwhocodes/config-array": { "node_modules/@humanwhocodes/config-array": {
"version": "0.11.11", "version": "0.11.11",
...@@ -10583,11 +10586,6 @@ ...@@ -10583,11 +10586,6 @@
"resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.3.tgz", "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.3.tgz",
"integrity": "sha512-0TNiGstbQmCFwt4akjjBg5pLRTSyj/PkWQ1ZoO2zntmg9yLqSRxwEa4iCfQLGjqhiqBfOJa7W/E8wfGrTDmlZQ==" "integrity": "sha512-0TNiGstbQmCFwt4akjjBg5pLRTSyj/PkWQ1ZoO2zntmg9yLqSRxwEa4iCfQLGjqhiqBfOJa7W/E8wfGrTDmlZQ=="
}, },
"node_modules/classnames": {
"version": "2.3.2",
"resolved": "https://registry.npmjs.org/classnames/-/classnames-2.3.2.tgz",
"integrity": "sha512-CSbhY4cFEJRe6/GQzIk5qXZ4Jeg5pcsP7b5peFSDpffpe1cqjASH/n9UTjBwOp6XpMSTwQ8Za2K5V02ueA7Tmw=="
},
"node_modules/clean-css": { "node_modules/clean-css": {
"version": "5.3.2", "version": "5.3.2",
"resolved": "https://registry.npmjs.org/clean-css/-/clean-css-5.3.2.tgz", "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-5.3.2.tgz",
...@@ -10700,6 +10698,14 @@ ...@@ -10700,6 +10698,14 @@
"node": ">=0.10.0" "node": ">=0.10.0"
} }
}, },
"node_modules/clsx": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz",
"integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==",
"engines": {
"node": ">=6"
}
},
"node_modules/co": { "node_modules/co": {
"version": "4.6.0", "version": "4.6.0",
"resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz",
...@@ -21549,17 +21555,16 @@ ...@@ -21549,17 +21555,16 @@
} }
} }
}, },
"node_modules/react-tooltip": { "node_modules/react-toastify": {
"version": "5.21.5", "version": "9.1.3",
"resolved": "https://registry.npmjs.org/react-tooltip/-/react-tooltip-5.21.5.tgz", "resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-9.1.3.tgz",
"integrity": "sha512-ey70qf6pBGi4U6xpyNlZAHobAhlo2dfxmImR2Bzd/DbLTsAYWz3TEaK+RMFuUZMq6hSPRbUHQSkP2rHBq4uFVg==", "integrity": "sha512-fPfb8ghtn/XMxw3LkxQBk3IyagNpF/LIKjOBflbexr2AWxAH1MJgvnESwEwBn9liLFXgTKWgBSdZpw9m4OTHTg==",
"dependencies": { "dependencies": {
"@floating-ui/dom": "^1.0.0", "clsx": "^1.1.1"
"classnames": "^2.3.0"
}, },
"peerDependencies": { "peerDependencies": {
"react": ">=16.14.0", "react": ">=16",
"react-dom": ">=16.14.0" "react-dom": ">=16"
} }
}, },
"node_modules/read-cache": { "node_modules/read-cache": {
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
"react-dom": "^18.2.0", "react-dom": "^18.2.0",
"react-router-dom": "^6.16.0", "react-router-dom": "^6.16.0",
"react-scripts": "5.0.1", "react-scripts": "5.0.1",
"react-toastify": "^9.1.3",
"web-vitals": "^2.1.4" "web-vitals": "^2.1.4"
}, },
"scripts": { "scripts": {
......
...@@ -24,7 +24,7 @@ ...@@ -24,7 +24,7 @@
work correctly both with client-side routing and a non-root public URL. work correctly both with client-side routing and a non-root public URL.
Learn how to configure a non-root public URL by running `npm run build`. Learn how to configure a non-root public URL by running `npm run build`.
--> -->
<title>React App</title> <title>Seat Booking</title>
</head> </head>
<body> <body>
<noscript>You need to enable JavaScript to run this app.</noscript> <noscript>You need to enable JavaScript to run this app.</noscript>
......
import { createContext, useContext, useState } from "react"; import { createContext, useContext, useState } from "react";
import * as api from "../lib/api"; import * as api from "../lib/api";
import { toast } from "react-toastify";
export const AppContext = createContext(); export const AppContext = createContext();
export const useAppContext = () => useContext(AppContext); export const useAppContext = () => useContext(AppContext);
...@@ -12,15 +13,14 @@ export function ContextProvider({ children }) { ...@@ -12,15 +13,14 @@ export function ContextProvider({ children }) {
setIsLoggedIn(true); setIsLoggedIn(true);
return response; return response;
} catch (err) { } catch (err) {
console.log(err.message); toast.error(err.message);
} }
}; };
const AddOrUpdateSeats = async ({ userId, seats, bookingId }) => { const AddOrUpdateSeats = async ({ userId, seats }) => {
try { try {
const response = await api.AddOrUpdateSeats({ const response = await api.AddOrUpdateSeats({
userId, userId,
seats, seats,
bookingId,
}); });
if (response.isAddSeat) { if (response.isAddSeat) {
setUserSeats([...userseats, response.response]); setUserSeats([...userseats, response.response]);
...@@ -31,7 +31,7 @@ export function ContextProvider({ children }) { ...@@ -31,7 +31,7 @@ export function ContextProvider({ children }) {
setUserSeats(result); setUserSeats(result);
} }
} catch (err) { } catch (err) {
console.log(err.message); toast.error(err.message);
} }
}; };
const GetAllReservedSeats = async () => { const GetAllReservedSeats = async () => {
...@@ -39,7 +39,7 @@ export function ContextProvider({ children }) { ...@@ -39,7 +39,7 @@ export function ContextProvider({ children }) {
const response = await api.GetAllSeats(); const response = await api.GetAllSeats();
setUserSeats(response); setUserSeats(response);
} catch (err) { } catch (err) {
console.log(err.message); toast.error(err.message);
} }
}; };
const value = { const value = {
......
...@@ -14,7 +14,8 @@ ...@@ -14,7 +14,8 @@
.card { .card {
display: block; display: block;
max-width: 30em; min-width: 30em;
max-width: 40dvw;
border-radius: 10px; border-radius: 10px;
min-height: 15em; min-height: 15em;
background-color: white; background-color: white;
......
.confirmation { .confirmation {
text-align: center; text-align: center;
padding: 1.5rem 3rem 1rem 3rem; padding: 1.5rem 2rem 1rem 2rem;
margin: 1.5rem 3rem 1rem 3rem; margin: 1.5rem 2rem 1rem 2rem;
} }
.confirmation img { .confirmation img {
......
...@@ -9,5 +9,6 @@ export default { ...@@ -9,5 +9,6 @@ export default {
export const confirmation = { export const confirmation = {
args: { args: {
seats: ["A1", "A2", "A3"], seats: ["A1", "A2", "A3"],
seatCount: 3,
}, },
}; };
...@@ -3,7 +3,7 @@ import Image from "../../../assets/conform.png"; ...@@ -3,7 +3,7 @@ import Image from "../../../assets/conform.png";
import style from "./confirmation.module.css"; import style from "./confirmation.module.css";
import Button from "../../Base/button"; import Button from "../../Base/button";
function Confirmation({ id, seatCount = 3, seats, onEdit, ...props }) { function Confirmation({ seatCount, seats, onEdit }) {
return ( return (
<Layout title="Booking Confirmed"> <Layout title="Booking Confirmed">
<div className={style.confirmation}> <div className={style.confirmation}>
......
...@@ -3,15 +3,17 @@ import Button from "../../Base/button"; ...@@ -3,15 +3,17 @@ import Button from "../../Base/button";
import Input from "../../Base/input"; import Input from "../../Base/input";
import Layout from "../../Base/layout"; import Layout from "../../Base/layout";
import styles from "./login.module.css"; import styles from "./login.module.css";
import { toast } from "react-toastify";
function Login({ onSubmit }) { function Login({ onSubmit }) {
const [mobile, setMobile] = useState(""); const [mobile, setMobile] = useState("");
// const [password, setPassword] = useState("");
const handleSubmit = (e) => { const handleSubmit = (e) => {
e.preventDefault(); e.preventDefault();
onSubmit({ mobile }); if (mobile.length < 10) {
setMobile(""); toast.info("mobile should be 10 digits");
//setPassword(""); } else {
onSubmit({ mobile });
setMobile("");
}
}; };
return ( return (
...@@ -30,14 +32,6 @@ function Login({ onSubmit }) { ...@@ -30,14 +32,6 @@ function Login({ onSubmit }) {
maxLength={10} maxLength={10}
onChange={(e) => setMobile(e.target.value)} onChange={(e) => setMobile(e.target.value)}
/> />
{/* <label>Password:</label>
<Input
placeholder="Enter Password"
required
type="password"
maxLength={8}
onChange={(e) => setPassword(e.target.value)}
/> */}
<Button>Submit</Button> <Button>Submit</Button>
</form> </form>
</Layout> </Layout>
......
...@@ -3,6 +3,7 @@ label { ...@@ -3,6 +3,7 @@ label {
} }
form { form {
padding: 1.5rem; padding: 1.5rem;
> * { > * {
inset: 0; inset: 0;
margin: 10px auto; margin: 10px auto;
......
...@@ -4,7 +4,7 @@ import Button from "../../Base/button"; ...@@ -4,7 +4,7 @@ import Button from "../../Base/button";
import styles from "./seat-layout.module.css"; import styles from "./seat-layout.module.css";
import Seat from "../seat"; import Seat from "../seat";
import Layout from "../../Base/layout"; import Layout from "../../Base/layout";
import { toast } from "react-toastify";
function SeatLayout({ function SeatLayout({
seats, seats,
reservedSeats, reservedSeats,
...@@ -18,8 +18,10 @@ function SeatLayout({ ...@@ -18,8 +18,10 @@ function SeatLayout({
const handleSeats = (seat) => { const handleSeats = (seat) => {
if (!selectedseat?.includes(seat) && selectedseat.length < seatLimit) { if (!selectedseat?.includes(seat) && selectedseat.length < seatLimit) {
setSelectedSeat([...selectedseat, seat]); setSelectedSeat([...selectedseat, seat]);
} else { } else if (selectedseat?.includes(seat)) {
setSelectedSeat(selectedseat?.filter((e) => e != seat)); setSelectedSeat(selectedseat?.filter((e) => e != seat));
} else {
toast.info(`Cannot select seat more than ${seatLimit}`);
} }
}; };
useEffect(() => { useEffect(() => {
......
...@@ -2,6 +2,7 @@ import { useState } from "react"; ...@@ -2,6 +2,7 @@ import { useState } from "react";
import Button from "../../Base/button"; import Button from "../../Base/button";
import Layout from "../../Base/layout"; import Layout from "../../Base/layout";
import styles from "./seatlimit.module.css"; import styles from "./seatlimit.module.css";
import Input from "../../Base/input";
function SeatLimit({ onLimit, ...props }) { function SeatLimit({ onLimit, ...props }) {
const [limit, setLimit] = useState(0); const [limit, setLimit] = useState(0);
...@@ -13,12 +14,17 @@ function SeatLimit({ onLimit, ...props }) { ...@@ -13,12 +14,17 @@ function SeatLimit({ onLimit, ...props }) {
<form onSubmit={handleLimit} className={styles["form"]}> <form onSubmit={handleLimit} className={styles["form"]}>
<div> <div>
<label>No of seats:</label> <label>No of seats:</label>
<input <Input
className={styles.input}
type="number" type="number"
value={limit} value={limit}
onChange={(e) => setLimit(e.target.value)} onChange={(e) => setLimit(e.target.value)}
/> />
{/* <input
className={styles.input}
type="number"
value={limit}
onChange={(e) => setLimit(e.target.value)}
/> */}
</div> </div>
<Button {...props}>Book Seats</Button> <Button {...props}>Book Seats</Button>
</form> </form>
......
...@@ -5,12 +5,26 @@ import App from "./App"; ...@@ -5,12 +5,26 @@ import App from "./App";
import reportWebVitals from "./reportWebVitals"; import reportWebVitals from "./reportWebVitals";
import { ContextProvider } from "./_context"; import { ContextProvider } from "./_context";
import { BrowserRouter as Router } from "react-router-dom"; import { BrowserRouter as Router } from "react-router-dom";
import { ToastContainer } from "react-toastify";
import "react-toastify/dist/ReactToastify.css";
const root = ReactDOM.createRoot(document.getElementById("root")); const root = ReactDOM.createRoot(document.getElementById("root"));
root.render( root.render(
<ContextProvider> <ContextProvider>
<Router> <Router>
<App /> <App />
</Router> </Router>
<ToastContainer
position="top-right"
autoClose={3000}
hideProgressBar={false}
newestOnTop={false}
closeOnClick
rtl={false}
pauseOnFocusLoss
draggable
pauseOnHover
theme="colored"
/>
</ContextProvider> </ContextProvider>
); );
......
import { toast } from "react-toastify";
const BASE_URL = "http://192.168.1.91:5000"; const BASE_URL = "http://192.168.1.91:5000";
export async function LoginOrRegister({ mobile }) { export async function LoginOrRegister({ mobile }) {
...@@ -13,7 +14,7 @@ export async function LoginOrRegister({ mobile }) { ...@@ -13,7 +14,7 @@ export async function LoginOrRegister({ mobile }) {
const user = await fetch(`${BASE_URL}/users`, { const user = await fetch(`${BASE_URL}/users`, {
method: "POST", method: "POST",
headers: { "Content-Type": "application/json" }, headers: { "Content-Type": "application/json" },
body: JSON.stringify({ mobile, isLogged: true }), body: JSON.stringify({ mobile }),
}); });
let _userData = [{ ...user }]; let _userData = [{ ...user }];
return _userData; return _userData;
...@@ -22,7 +23,7 @@ export async function LoginOrRegister({ mobile }) { ...@@ -22,7 +23,7 @@ export async function LoginOrRegister({ mobile }) {
return result; return result;
} }
} catch (err) { } catch (err) {
throw err; // Rethrow the error for the caller to handle toast.error(err.message);
} }
} }
...@@ -62,7 +63,7 @@ export async function AddOrUpdateSeats({ userId, seats }) { ...@@ -62,7 +63,7 @@ export async function AddOrUpdateSeats({ userId, seats }) {
}; };
} }
} catch (err) { } catch (err) {
throw err; toast.error(err.message);
} }
} }
...@@ -71,6 +72,6 @@ export async function GetAllSeats() { ...@@ -71,6 +72,6 @@ export async function GetAllSeats() {
const response = await fetch(BASE_URL + "/reservedSeats"); const response = await fetch(BASE_URL + "/reservedSeats");
return await response.json(); return await response.json();
} catch (err) { } catch (err) {
throw err; toast.error(err.message);
} }
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment