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

Deleting users from teams #27

Merged
merged 12 commits into from
Mar 25, 2025
Merged
27 changes: 0 additions & 27 deletions auth.config.ts

This file was deleted.

9 changes: 2 additions & 7 deletions components/app-sidebar.tsx
Original file line number Diff line number Diff line change
@@ -1,8 +1,6 @@
import {
Inbox,
Users,
User,
UserPlus,
Files,
GitPullRequest,
LogOut,
Expand All @@ -23,10 +21,8 @@ import {
SidebarMenuButton,
SidebarMenuItem,
SidebarHeader,
SidebarFooter,
} from "@/components/ui/sidebar";
import SignOutModal from "@/components/auth/sign-out-modal";
import { NetworkStatusIndicator } from "@/components/ui/network-status-indicator";
import { useRouter } from "next/router";

const data = {
Expand Down Expand Up @@ -59,8 +55,7 @@ const data = {
icon: Files,
},
{
title: "Logout",
url: "/api/auth/signout",
title: "Sign Out",
icon: LogOut,
},
],
Expand Down Expand Up @@ -115,7 +110,7 @@ export function AppSidebar() {
<SidebarMenu>
{item.items.map((subItem) => (
<SidebarMenuItem key={subItem.title}>
{subItem.title === "Logout" ? (
{subItem.title === "Sign Out" ? (
<SignOutModal>
<SidebarMenuButton asChild>
<button className="flex w-full items-center">
Expand Down
111 changes: 0 additions & 111 deletions components/auth/login-form.tsx

This file was deleted.

135 changes: 135 additions & 0 deletions components/auth/sign-in-form.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
"use client";

import {
AtSymbolIcon,
KeyIcon,
ExclamationCircleIcon,
} from "@heroicons/react/24/outline";
import { ArrowRightIcon } from "@heroicons/react/20/solid";
import { Button } from "../ui/button";
import { useState } from "react";
import { useRouter } from "next/router";
import { toast } from "react-hot-toast";
import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import { schemaSignIn } from "@/lib/schemas/sign-in.schema";
import { z } from "zod";
import { signIn } from "next-auth/react";

type FormData = z.infer<typeof schemaSignIn>;

export function SignInForm() {
const [isLoading, setIsLoading] = useState(false);
const router = useRouter();
const form = useForm<FormData>({
resolver: zodResolver(schemaSignIn),
defaultValues: {
email: "",
password: "",
},
});

const onSubmit = async (data: FormData) => {
try {
setIsLoading(true);

const result = await signIn("credentials", {
email: data.email,
password: data.password,
redirect: false,
callbackUrl: "/teams",
});

if (result?.error) {
toast.error("Invalid email or password");
return;
}
toast.success("Sign in successful!");
router.push("/teams");
} catch (error) {
toast.error(
error instanceof Error ? error.message : "Something went wrong"
);
console.error("Sign in error:", error);
} finally {
setIsLoading(false);
}
};

return (
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-3">
<div className="flex-1 rounded-lg bg-gray-50 px-6 pb-4 pt-8">
<h1 className={`mb-3 text-2xl`}>Please sign in to continue.</h1>
<div className="w-full">
<div>
<label
className="mb-3 mt-5 block text-xs font-medium text-gray-900"
htmlFor="email"
>
Email
</label>
<div className="relative">
<input
className="peer block w-full rounded-md border border-gray-200 py-[9px] pl-10 text-sm outline-2 placeholder:text-gray-500"
id="email"
type="email"
{...form.register("email")}
placeholder="Enter your email address"
aria-describedby="email-error"
/>
<AtSymbolIcon className="pointer-events-none absolute left-3 top-1/2 h-[18px] w-[18px] -translate-y-1/2 text-gray-500 peer-focus:text-gray-900" />
</div>
{form.formState.errors.email && (
<div className="mt-2 flex items-center text-sm text-red-500">
<ExclamationCircleIcon className="mr-1 h-4 w-4" />
<p>{form.formState.errors.email.message}</p>
</div>
)}
</div>
<div className="mt-4">
<label
className="mb-3 mt-5 block text-xs font-medium text-gray-900"
htmlFor="password"
>
Password
</label>
<div className="relative">
<input
className="peer block w-full rounded-md border border-gray-200 py-[9px] pl-10 text-sm outline-2 placeholder:text-gray-500"
id="password"
type="password"
{...form.register("password")}
placeholder="Enter password"
aria-describedby="password-error"
/>
<KeyIcon className="pointer-events-none absolute left-3 top-1/2 h-[18px] w-[18px] -translate-y-1/2 text-gray-500 peer-focus:text-gray-900" />
</div>
{form.formState.errors.password && (
<div className="mt-2 flex items-center text-sm text-red-500">
<ExclamationCircleIcon className="mr-1 h-4 w-4" />
<p>{form.formState.errors.password.message}</p>
</div>
)}
</div>
</div>
<Button
type="submit"
className="mt-6 w-full"
aria-disabled={isLoading}
disabled={isLoading}
>
{isLoading ? (
<div className="flex items-center justify-center gap-2">
<div className="h-5 w-5 animate-spin rounded-full border-2 border-white border-t-transparent" />
Signing in...
</div>
) : (
<div className="flex items-center justify-center gap-2">
Sign in <ArrowRightIcon className="ml-auto h-5 w-5" />
</div>
)}
</Button>
</div>
</form>
);
}
4 changes: 2 additions & 2 deletions components/auth/sign-out-modal.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,10 @@ const SignOutModal = ({ children }: SignOutModalProps) => {

await signOut({
redirect: false,
callbackUrl: "/auth/login",
callbackUrl: "/auth/sign-in",
});

router.push("/auth/login");
router.push("/auth/sign-in");
} catch (error) {
console.error("Error during sign out:", error);
} finally {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,17 +1,12 @@
import {
AtSymbolIcon,
KeyIcon,
ExclamationCircleIcon,
UserIcon,
} from "@heroicons/react/24/outline";
import { AtSymbolIcon, KeyIcon, UserIcon } from "@heroicons/react/24/outline";
import { ArrowRightIcon } from "@heroicons/react/20/solid";
import { Button } from "../ui/button";
import { register } from "@/lib/auth/actions";
import { signUp } from "@/lib/auth/actions";
import { useState } from "react";
import { useRouter } from "next/router";
import toast from "react-hot-toast";

export function RegisterForm() {
export function SignUpForm() {
const [isLoading, setIsLoading] = useState(false);
const router = useRouter();

Expand All @@ -21,7 +16,7 @@ export function RegisterForm() {
const formData = new FormData(e.currentTarget);

try {
const { error } = await register(formData);
const { error } = await signUp(formData);

if (error) {
toast.error(error);
Expand All @@ -30,7 +25,7 @@ export function RegisterForm() {
}

toast.success("Account created successfully! Please log in.");
await router.push("/auth/login");
await router.push("/auth/sign-in");
} catch (error) {
toast.error("An error occurred while creating your account");
setIsLoading(false);
Expand Down
2 changes: 1 addition & 1 deletion components/board/board.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import { ShapeColorPalette } from "@/components/board/components/shape-color-pal
import { KonvaEventObject } from "konva/lib/Node";
import { Text } from "konva/lib/shapes/Text";

export default function Board({}: BoardProps) {
export default function Board() {
const clientSyncService = useClientSync();
const docUrl = clientSyncService.getDocUrl() as AnyDocumentId;
const [localDoc] = useDocument<LayerSchema>(docUrl);
Expand Down
5 changes: 2 additions & 3 deletions components/board/hooks/use-board-interactions.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { useCallback, useContext, useRef, useEffect } from "react";
import { useCallback, useContext, useRef } from "react";
import { KonvaEventObject } from "konva/lib/Node";
import { BoardMode } from "@/types/board";
import { useDrawing } from "./use-drawing";
import { useDragging } from "./use-dragging";
import { useErasing } from "./use-erasing";
Expand Down Expand Up @@ -57,7 +56,7 @@ export const useBoardInteractions = () => {
(e: KeyboardEvent) => {
if (mode !== "selecting") return;

if (e.key === "Delete" || e.key === "Backspace") {
if (e.key === "Delete") {
e.preventDefault();
handleDelete();
}
Expand Down
Loading