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

Toasts (Notification Component) #163

Merged
merged 30 commits into from
May 26, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
2d89ec0
feature: responsive navbar
adityanandanx May 24, 2023
ded68b7
style: nav links hover animatiosn
adityanandanx May 24, 2023
6e11578
Merge branch 'main' into mad/ui-improvements
adityanandanx May 25, 2023
df116aa
style: better Input Fields
adityanandanx May 25, 2023
6d5188e
refactor: use form submit instead of button onclick
adityanandanx May 25, 2023
264bfd5
feature: loading states
adityanandanx May 25, 2023
6fdb8ef
feature: log out confirmation
adityanandanx May 25, 2023
791aaf0
feature: basic toast
adityanandanx May 25, 2023
3fb0023
feature: Toast variants
adityanandanx May 25, 2023
380e3a5
fix: use global toast provider
adityanandanx May 25, 2023
19c17f6
feature: use toast instead of alert for auth routes
adityanandanx May 25, 2023
67a3726
fix(mobile): nav menu close on route change
adityanandanx May 25, 2023
f171f5e
fix: field dark mode
adityanandanx May 25, 2023
e00a853
feature: redirect when login and logout
adityanandanx May 25, 2023
0d464f0
refactor: group auth routes
adityanandanx May 25, 2023
6c2c34e
refactor: use @/app imports
adityanandanx May 25, 2023
74464be
Merge branch 'main' into mad/ui-improvements
adityanandanx May 25, 2023
27d5d4e
style: use Field on /upload
adityanandanx May 25, 2023
a892bf4
fix: forward ref
adityanandanx May 25, 2023
25c2948
feature: Multi toast
adityanandanx May 25, 2023
9225847
feature: add toasts to /upload
adityanandanx May 26, 2023
859cfab
Merge branch 'main' into mad/ui-improvements
adityanandanx May 26, 2023
1566008
refactor: new login in auth group
adityanandanx May 26, 2023
20f3f2c
chore: quote
adityanandanx May 26, 2023
6ed8337
chore(pnpm): removed
StanGirard May 26, 2023
339ca68
Merge branch 'main' into mad/ui-improvements
adityanandanx May 26, 2023
2a45a31
feature: toasty animations
adityanandanx May 26, 2023
ebbadb8
fix: build errors and warnings
adityanandanx May 26, 2023
fbee42f
chore: remove irrelevant comments
adityanandanx May 26, 2023
3e64c22
fix: use unique ids for toasts
adityanandanx May 26, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -50,4 +50,4 @@ streamlit-demo/.streamlit/secrets.toml
.frontend_env
backend/pandoc-*
**/yarn.lock
**/.pandoc-*
**/.pandoc-*
83 changes: 83 additions & 0 deletions frontend/app/(auth)/login/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
"use client";
import { useRef, useState } from "react";
import { useRouter } from "next/navigation";
import { useSupabase } from "@/app/supabase-provider";
import Button from "@/app/components/ui/Button";
import Card from "@/app/components/ui/Card";
import Field from "@/app/components/ui/Field";
import PageHeading from "@/app/components/ui/PageHeading";
import Toast, { ToastRef } from "@/app/components/ui/Toast";
import Link from "next/link";

export default function Login() {
const { supabase } = useSupabase();
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [isPending, setIsPending] = useState(false);

const loginToast = useRef<ToastRef>(null);

const router = useRouter();

const handleLogin = async () => {
setIsPending(true);
const { data, error } = await supabase.auth.signInWithPassword({
email: email,
password: password,
});

if (error) {
console.error("Error logging in:", error.message);
loginToast.current?.publish({
variant: "danger",
text: error.message,
});
} else if (data) {
console.log("User logged in:", data);
loginToast.current?.publish({
variant: "success",
text: "Successfully logged in",
});
router.replace("/");
}
setIsPending(false);
};

return (
<main>
<section className="w-full min-h-screen h-full outline-none flex flex-col gap-5 items-center justify-center p-6">
<PageHeading title="Login" subtitle="Welcome back" />
<Card className="max-w-md w-full p-5 sm:p-10 text-left">
<form
onSubmit={(e) => {
e.preventDefault();
handleLogin();
}}
className="flex flex-col gap-2"
>
<Field
name="email"
required
type="email"
placeholder="Email"
onChange={(e) => setEmail(e.target.value)}
/>
<Field
name="password"
required
type="password"
value={password}
onChange={(e) => setPassword(e.target.value)}
placeholder="Password"
/>
<div className="flex flex-col items-center justify-center mt-2 gap-2">
<Button isLoading={isPending}>Login</Button>
<Link href="/signup">Don{"'"}t have an account? Sign up</Link>
</div>
</form>
</Card>
</section>
<Toast variant="success" ref={loginToast} />
</main>
);
}
74 changes: 74 additions & 0 deletions frontend/app/(auth)/logout/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
"use client";
import { useEffect, useRef, useState } from "react";
import { useRouter } from "next/navigation";
import { useSupabase } from "@/app/supabase-provider";
import Toast, { ToastRef } from "@/app/components/ui/Toast";
import PageHeading from "@/app/components/ui/PageHeading";
import Button from "@/app/components/ui/Button";
import Card from "@/app/components/ui/Card";
import Link from "next/link";

export default function Logout() {
const { supabase } = useSupabase();
const [isPending, setIsPending] = useState(false);

const logoutToast = useRef<ToastRef>(null);
const [error, setError] = useState("Unknown Error");

const router = useRouter();

const handleLogout = async () => {
setIsPending(true);
const { error } = await supabase.auth.signOut();

if (error) {
console.error("Error logging out:", error.message);
setError(error.message);
logoutToast.current?.publish({
variant: "danger",
text: `Error logging out: ${error.message}`,
});
} else {
console.log("User logged out");
logoutToast.current?.publish({
variant: "success",
text: "Logged out successfully",
});
router.replace("/");
}
setIsPending(false);
};

// useEffect(() => {
// handleLogout();
// }, []);

return (
<main>
<section className="w-full min-h-screen h-full outline-none flex flex-col gap-5 items-center justify-center p-6">
<PageHeading title="Logout" subtitle="See you next time" />
<Card className="max-w-md w-full p-5 sm:p-10 text-center flex flex-col items-center gap-5">
<h2 className="text-lg">Are you sure you want to sign out?</h2>
<div className="flex gap-5 items-center justify-center">
<Link href={"/"}>
<Button variant={"primary"}>Go back</Button>
</Link>
<Button
isLoading={isPending}
variant={"danger"}
onClick={() => handleLogout()}
>
Log Out
</Button>
</div>
</Card>
</section>
<Toast variant="success" ref={logoutToast}>
Logged Out Successfully
</Toast>
<Toast variant="danger" ref={logoutToast}>
{error}
</Toast>
</main>
);
}
78 changes: 78 additions & 0 deletions frontend/app/(auth)/signup/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
"use client";
import Button from "@/app/components/ui/Button";
import Card from "@/app/components/ui/Card";
import Field from "@/app/components/ui/Field";
import PageHeading from "@/app/components/ui/PageHeading";
import Toast, { ToastRef } from "@/app/components/ui/Toast";
import { useSupabase } from "@/app/supabase-provider";
import { useRef, useState } from "react";

export default function SignUp() {
const { supabase } = useSupabase();
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [isPending, setIsPending] = useState(false);

const signupToast = useRef<ToastRef>(null);

const handleSignUp = async () => {
setIsPending(true);
const { data, error } = await supabase.auth.signUp({
email: email,
password: password,
});

if (error) {
console.error("Error signing up:", error.message);
signupToast.current?.publish({
variant: "danger",
text: `Error signing up: ${error.message}`,
});
} else if (data) {
console.log("User signed up");
signupToast.current?.publish({ variant: "success", text: "Sign" });
}
setIsPending(false);
};

return (
<main>
<section className="w-full min-h-screen h-full outline-none flex flex-col gap-5 items-center justify-center p-6">
<PageHeading title="Sign Up" subtitle="Create your account" />
<Card className="max-w-md w-full p-5 sm:p-10 text-left">
<form
onSubmit={(e) => {
e.preventDefault();
handleSignUp();
}}
className="flex flex-col gap-2"
>
<Field
name="email"
required
type="email"
placeholder="Email"
onChange={(e) => setEmail(e.target.value)}
/>
<Field
name="password"
required
type="password"
value={password}
onChange={(e) => setPassword(e.target.value)}
placeholder="Password"
/>
<div className="flex flex-col items-center justify-center mt-2 gap-2">
<Button isLoading={isPending}>Sign Up</Button>
</div>
</form>
</Card>
</section>
<Toast variant="success" ref={signupToast}>
<h1 className="font-bold">Confirmation Email sent</h1>
<p className="text-sm">Check your email.</p>
</Toast>
<Toast variant="danger" ref={signupToast} />
</main>
);
}
8 changes: 4 additions & 4 deletions frontend/app/chat/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,9 @@ export default function ChatPage() {
const [temperature, setTemperature] = useState(0);
const [maxTokens, setMaxTokens] = useState(500);
const [isPending, setIsPending] = useState(false);
const { supabase, session } = useSupabase()
const { supabase, session } = useSupabase();
if (session === null) {
redirect('/login')
redirect("/login");
}

const askQuestion = async () => {
Expand All @@ -36,7 +36,7 @@ export default function ChatPage() {
},
{
headers: {
'Authorization': `Bearer ${session.access_token}`,
Authorization: `Bearer ${session.access_token}`,
},
}
);
Expand All @@ -48,7 +48,7 @@ export default function ChatPage() {
};

return (
<main className="min-h-screen w-full flex flex-col pt-20">
<main className="min-h-screen w-full flex flex-col pt-32">
<section className="flex flex-col justify-center items-center flex-1 gap-5 h-full">
<PageHeading
title="Chat with your brain"
Expand Down
7 changes: 5 additions & 2 deletions frontend/app/components/NavBar/MobileMenu.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ interface MobileMenuProps {}
const MobileMenu: FC<MobileMenuProps> = ({}) => {
const [open, setOpen] = useState(false);
return (
<Dialog.Root onOpenChange={setOpen}>
<Dialog.Root onOpenChange={setOpen} open={open}>
<Dialog.Trigger asChild>
<button className="block sm:hidden" aria-label="open menu">
<MdMenu className="text-4xl" />
Expand All @@ -29,7 +29,10 @@ const MobileMenu: FC<MobileMenuProps> = ({}) => {
>
<Dialog.Content asChild forceMount>
<div className="flex flex-col items-center justify-between py-24 flex-1 w-full bg-white dark:bg-black border border-black/10 dark:border-white/25 p-10 shadow-xl dark:shadow-primary/50 focus:outline-none cursor-auto z-50">
<NavItems className="text-3xl h-fit text-center flex-col items-center justify-center gap-10" />
<NavItems
setOpen={setOpen}
className="text-3xl h-fit text-center flex-col items-center justify-center gap-10"
/>

<p className="">
Get a Second Brain with{" "}
Expand Down
37 changes: 25 additions & 12 deletions frontend/app/components/NavBar/NavItems.tsx
Original file line number Diff line number Diff line change
@@ -1,14 +1,15 @@
import { cn } from "@/lib/utils";
import Link from "next/link";
import { FC, HTMLAttributes, ReactNode } from "react";
import { Dispatch, FC, HTMLAttributes, ReactNode, SetStateAction } from "react";
import DarkModeToggle from "./DarkModeToggle";
import Button from "../ui/Button";

interface NavItemsProps extends HTMLAttributes<HTMLUListElement> {}
interface NavItemsProps extends HTMLAttributes<HTMLUListElement> {
setOpen?: Dispatch<SetStateAction<boolean>>;
}

const NavItems: FC<NavItemsProps> = ({ className, ...props }) => {
const NavItems: FC<NavItemsProps> = ({ className, setOpen, ...props }) => {
return (
// <div className={cn("flex flex-1 items-center", className)} {...props}>
<ul
className={cn(
"flex flex-row items-center gap-4 text-sm flex-1",
Expand All @@ -18,14 +19,24 @@ const NavItems: FC<NavItemsProps> = ({ className, ...props }) => {
>
{process.env.NEXT_PUBLIC_ENV === "local" ? (
<>
<NavLink to="/upload">Upload</NavLink>
<NavLink to="/chat">Chat</NavLink>
<NavLink to="/explore">Explore</NavLink>
<NavLink setOpen={setOpen} to="/upload">
Upload
</NavLink>
<NavLink setOpen={setOpen} to="/chat">
Chat
</NavLink>
<NavLink setOpen={setOpen} to="/explore">
Explore
</NavLink>
</>
) : (
<>
<NavLink to="https://github.com/StanGirard/quivr">Github</NavLink>
<NavLink to="https://discord.gg/HUpRgp2HG8">Discord</NavLink>
<NavLink setOpen={setOpen} to="https://github.com/StanGirard/quivr">
Github
</NavLink>
<NavLink setOpen={setOpen} to="https://discord.gg/HUpRgp2HG8">
Discord
</NavLink>
</>
)}
<div className="flex sm:flex-1 sm:justify-end flex-col items-center justify-center sm:flex-row gap-5 sm:gap-2">
Expand All @@ -35,19 +46,21 @@ const NavItems: FC<NavItemsProps> = ({ className, ...props }) => {
<DarkModeToggle />
</div>
</ul>
// </div>
);
};

interface NavLinkProps {
children: ReactNode;
to: string;
setOpen?: Dispatch<SetStateAction<boolean>>;
}

const NavLink: FC<NavLinkProps> = ({ children, to }) => {
const NavLink: FC<NavLinkProps> = ({ children, to, setOpen }) => {
return (
<li className="group relative">
<Link href={to}>{children}</Link>
<Link onClick={() => setOpen && setOpen(false)} href={to}>
{children}
</Link>
<hr className="aboslute top-full border border-transparent border-b-primary dark:border-b-white scale-x-0 group-hover:scale-x-100 group-focus-within:scale-x-100 transition-transform" />
</li>
);
Expand Down
2 changes: 1 addition & 1 deletion frontend/app/components/NavBar/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ const NavBar: FC<NavBarProps> = ({}) => {

window.addEventListener("scroll", handleScroll);
return () => window.removeEventListener("scroll", handleScroll);
}, []);
}, [hidden]);

return (
<motion.header
Expand Down
2 changes: 1 addition & 1 deletion frontend/app/components/ui/Button.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ const ButtonVariants = cva(
variants: {
variant: {
primary:
"bg-black disabled:bg-gray-500 disabled:hover:bg-gray-500 text-white dark:bg-white dark:text-black hover:bg-gray-700 dark:hover:bg-gray-200 transition-colors",
"bg-black border border-black dark:border-white disabled:bg-gray-500 disabled:hover:bg-gray-500 text-white dark:bg-white dark:text-black hover:bg-gray-700 dark:hover:bg-gray-200 transition-colors",
tertiary: "text-black dark:text-white bg-transparent py-2 px-4",
secondary:
"border border-black dark:border-white bg-white dark:bg-black text-black dark:text-white focus:bg-black dark:focus:bg-white hover:bg-black dark:hover:bg-white hover:text-white dark:hover:text-black focus:text-white dark:focus:text-black transition-colors py-2 px-4 shadow-none",
Expand Down
Loading