diff --git a/src/assets/encrypted-data-illustration.svg b/src/assets/encrypted-data-illustration.svg new file mode 100644 index 0000000..c8032e7 --- /dev/null +++ b/src/assets/encrypted-data-illustration.svg @@ -0,0 +1,13 @@ + + + + + + + + + + + + + diff --git a/src/assets/success-illustration.svg b/src/assets/success-illustration.svg new file mode 100644 index 0000000..7f2d309 --- /dev/null +++ b/src/assets/success-illustration.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/components/DocumentUpload.tsx b/src/components/DocumentUpload.tsx new file mode 100644 index 0000000..1dadd99 --- /dev/null +++ b/src/components/DocumentUpload.tsx @@ -0,0 +1,138 @@ +import { forwardRef, useState } from "react"; +import { UploadIcon, CheckCircle, Trash } from "lucide-react"; +import { cn } from "@/lib/utils"; +import { formatFileSize, truncateFileName } from "@/utils"; + +export type DocumentUploadProps = React.InputHTMLAttributes; + +const DocumentUpload = forwardRef( + ({ className, id, ...props }, ref) => { + const [isFocused, setIsFocused] = useState(false); + const [fileName, setFileName] = useState(null); + const [fileSize, setFileSize] = useState(null); + const [progress, setProgress] = useState(0); + + const handleFileChange = (event: React.ChangeEvent) => { + const file = event.target.files?.[0]; + + if (file) { + setFileName(truncateFileName(file.name)); + setFileSize(file.size); + setProgress(0); + uploadFile(file); + } + }; + + const clearFileSelection = () => { + setFileName(null); + setFileSize(null); + setProgress(0); + if (ref && typeof ref === "object" && ref.current) { + ref.current.value = ""; + } + }; + + const uploadFile = async (file: File) => { + const formData = new FormData(); + formData.append("file", file); + + // todo: replace the endpoint with the actual upload endpoint + const response = await fetch("/upload-endpoint", { + method: "POST", + body: formData, + }); + + if (response.ok) { + console.log("Upload complete"); + setProgress(100); + } else { + console.error("Upload failed"); + } + }; + + return ( +
+
+ setIsFocused(true)} + onBlur={() => setIsFocused(false)} + onChange={handleFileChange} + {...props} + /> + +
+
+ +
+
+ + Upload your identity document + + + PNG or JPG (Max. 5mb) + +
+
+
+ + {fileName && ( +
+
+
+
+ + {fileName} + + + + {formatFileSize(fileSize || 0)} + +
+ + +
+ + +
+ +
+
+
+
+ {progress}% +
+
+ )} +
+ ); + } +); + +DocumentUpload.displayName = "DocumentUpload"; + +export { DocumentUpload }; diff --git a/src/components/ui/input.tsx b/src/components/ui/input.tsx index a921025..ea070ba 100644 --- a/src/components/ui/input.tsx +++ b/src/components/ui/input.tsx @@ -1,25 +1,70 @@ -import * as React from "react" - -import { cn } from "@/lib/utils" +import { forwardRef, useState } from "react"; +import { cn } from "@/lib/utils"; export interface InputProps - extends React.InputHTMLAttributes {} + extends React.InputHTMLAttributes { + label: string; + required?: boolean; +} + +const Input = forwardRef( + ({ className, type, label, required = false, id, ...props }, ref) => { + const [isFocused, setIsFocused] = useState(false); + const [hasValue, setHasValue] = useState(false); + + const handleBlur = (e: React.FocusEvent) => { + setIsFocused(false); + setHasValue(!!e.target.value); + }; + + const handleChange = (e: React.ChangeEvent) => { + setHasValue(!!e.target.value); + }; -const Input = React.forwardRef( - ({ className, type, ...props }, ref) => { return ( - - ) +
+ setIsFocused(true)} + onBlur={handleBlur} + onChange={handleChange} + {...props} + /> + +
+ ); } -) -Input.displayName = "Input" +); +Input.displayName = "Input"; -export { Input } +export { Input }; diff --git a/src/components/ui/select.tsx b/src/components/ui/select.tsx index fe56d4d..7db0aeb 100644 --- a/src/components/ui/select.tsx +++ b/src/components/ui/select.tsx @@ -1,148 +1,210 @@ -import * as React from "react" -import * as SelectPrimitive from "@radix-ui/react-select" -import { Check, ChevronDown, ChevronUp } from "lucide-react" +import * as React from "react"; +import * as SelectPrimitive from "@radix-ui/react-select"; +import { Check, ChevronDown, ChevronUp } from "lucide-react"; +import { cn } from "@/lib/utils"; -import { cn } from "@/lib/utils" +interface SelectProps { + children: React.ReactNode; + onValueChange?: (value: string) => void; +} + +const Select: React.FC = ({ children, onValueChange }) => { + const [hasValue, setHasValue] = React.useState(false); + const [triggerWidth, setTriggerWidth] = React.useState(null); + const [isOpen, setIsOpen] = React.useState(false); -const Select = SelectPrimitive.Root + const handleValueChange = (value: string) => { + setHasValue(!!value); + if (onValueChange) onValueChange(value); + }; -const SelectGroup = SelectPrimitive.Group + return ( + + {React.Children.map(children, (child) => { + if (React.isValidElement(child) && child.type === SelectTrigger) { + return React.cloneElement(child as React.ReactElement, { + hasValue, + setTriggerWidth, + isOpen, + }); + } + if (React.isValidElement(child) && child.type === SelectContent) { + return React.cloneElement(child as React.ReactElement, { + triggerWidth, + }); + } + return child; + })} + + ); +}; -const SelectValue = SelectPrimitive.Value +interface SelectTriggerProps + extends React.ComponentPropsWithoutRef { + label: string; + hasValue?: boolean; + setTriggerWidth?: (width: number) => void; + isOpen?: boolean; +} const SelectTrigger = React.forwardRef< React.ElementRef, - React.ComponentPropsWithoutRef ->(({ className, children, ...props }, ref) => ( - span]:line-clamp-1", - className - )} - {...props} - > - {children} - - - - -)) -SelectTrigger.displayName = SelectPrimitive.Trigger.displayName - -const SelectScrollUpButton = React.forwardRef< - React.ElementRef, - React.ComponentPropsWithoutRef ->(({ className, ...props }, ref) => ( - - - -)) -SelectScrollUpButton.displayName = SelectPrimitive.ScrollUpButton.displayName - -const SelectScrollDownButton = React.forwardRef< - React.ElementRef, - React.ComponentPropsWithoutRef ->(({ className, ...props }, ref) => ( - - - -)) -SelectScrollDownButton.displayName = - SelectPrimitive.ScrollDownButton.displayName + SelectTriggerProps +>(({ className, label, hasValue, setTriggerWidth, isOpen, ...props }, ref) => { + const triggerRef = React.useRef(null); + + React.useEffect(() => { + if (triggerRef.current && setTriggerWidth) { + setTriggerWidth(triggerRef.current.offsetWidth); + } + }, [setTriggerWidth]); + + return ( +
+ { + // @ts-expect-error This is a valid mutable property + triggerRef.current = node; + + if (typeof ref === "function") ref(node); + else if (ref) ref.current = node; + }} + className={cn( + "flex h-[58px] w-full items-center justify-between rounded-[8px] border border-[#1B0F0014] bg-elevation-200 px-4 text-sm transition-all duration-200 ease-in-out outline-none", + hasValue ? "pt-6 pb-2" : "pt-5 pb-3", + className + )} + {...props} + > +
+ +
+
+ + {isOpen ? ( + + ) : ( + + )} + +
+
+ +
+ ); +}); +SelectTrigger.displayName = SelectPrimitive.Trigger.displayName; + +interface SelectContentProps + extends React.ComponentPropsWithoutRef { + triggerWidth?: number | null; +} const SelectContent = React.forwardRef< React.ElementRef, - React.ComponentPropsWithoutRef ->(({ className, children, position = "popper", ...props }, ref) => ( - - - - ( + ( + { className, children, triggerWidth, position = "popper", ...props }, + ref + ) => ( + + - {children} - - - - -)) -SelectContent.displayName = SelectPrimitive.Content.displayName + + {children} + + + + ) +); +SelectContent.displayName = SelectPrimitive.Content.displayName; + +const SelectGroup = SelectPrimitive.Group; +const SelectValue = SelectPrimitive.Value; + +type SelectLabelProps = React.ComponentPropsWithoutRef< + typeof SelectPrimitive.Label +>; const SelectLabel = React.forwardRef< React.ElementRef, - React.ComponentPropsWithoutRef + SelectLabelProps >(({ className, ...props }, ref) => ( -)) -SelectLabel.displayName = SelectPrimitive.Label.displayName +)); +SelectLabel.displayName = SelectPrimitive.Label.displayName; + +type SelectItemProps = React.ComponentPropsWithoutRef< + typeof SelectPrimitive.Item +>; const SelectItem = React.forwardRef< React.ElementRef, - React.ComponentPropsWithoutRef + SelectItemProps >(({ className, children, ...props }, ref) => ( - + {children} + - + - - {children} -)) -SelectItem.displayName = SelectPrimitive.Item.displayName +)); +SelectItem.displayName = SelectPrimitive.Item.displayName; + +type SelectSeparatorProps = React.ComponentPropsWithoutRef< + typeof SelectPrimitive.Separator +>; const SelectSeparator = React.forwardRef< React.ElementRef, - React.ComponentPropsWithoutRef + SelectSeparatorProps >(({ className, ...props }, ref) => ( -)) -SelectSeparator.displayName = SelectPrimitive.Separator.displayName +)); +SelectSeparator.displayName = SelectPrimitive.Separator.displayName; export { Select, @@ -153,6 +215,4 @@ export { SelectLabel, SelectItem, SelectSeparator, - SelectScrollUpButton, - SelectScrollDownButton, -} +}; diff --git a/src/constants/routes.ts b/src/constants/routes.ts index f2edc36..aaf66c4 100644 --- a/src/constants/routes.ts +++ b/src/constants/routes.ts @@ -2,8 +2,11 @@ const LOGIN = "/login"; const UNLOCK = "/unlock"; const RESTORE_WITH_PRIVATE_KEY = "/private-key"; const RESTORE_WITH_SEED_PHRASE = "/seed-phrase"; - +const CREATE_IDENTITY = "/create-identity"; const HOME = "/home"; +const REQUIRED_INFORMATION = "/required-information"; +const EMAIL_VERIFICATION = "/email-verification"; +const OPTIONAL_INFORMATION = "/optional-information"; const NOTIFICATIONS = "/notifications"; export default { @@ -12,5 +15,9 @@ export default { HOME, RESTORE_WITH_PRIVATE_KEY, RESTORE_WITH_SEED_PHRASE, + CREATE_IDENTITY, + REQUIRED_INFORMATION, + EMAIL_VERIFICATION, + OPTIONAL_INFORMATION, NOTIFICATIONS, }; diff --git a/src/main.tsx b/src/main.tsx index efbac7e..fa30b14 100644 --- a/src/main.tsx +++ b/src/main.tsx @@ -5,19 +5,21 @@ import { createBrowserRouter, RouterProvider } from "react-router-dom"; import LanguageProvider from "./context/LanguageContext"; import DefaultLayout from "./layouts/Default"; -import CreateIdentity from "./pages/onboarding/CreateIdentity"; -import ConnectCompany from "./pages/onboarding/ConnectCompany"; -import ProfileInfo from "./pages/onboarding/ProfileInfo"; - +import RequiredInformation from "./pages/onboarding/RequiredInformation"; import Unlock from "./pages/Unlock"; import Login from "./pages/Login"; import RecoverWithSeedPhrase from "./pages/RecoverWithSeedPhrase"; import Home from "./pages/Home"; +import CreateNewIdentity from "./pages/onboarding/CreateNewIdentity"; +import Success from "./pages/onboarding/Success"; import { Notifications, NotificationsEmpty } from "./pages/Notifications"; import routes from "./constants/routes"; import "./index.css"; import "./styles/fonts.css"; +import EmailVerification from "./pages/onboarding/EmailVerification"; +import OptionalInformation from "./pages/onboarding/OptionalInformation"; +import ConfirmIdentity from "./pages/onboarding/ConfirmIdentity"; const router = createBrowserRouter([ { @@ -41,16 +43,28 @@ const router = createBrowserRouter([ element: , }, { - path: "/create-identity", - element: , + path: routes.CREATE_IDENTITY, + element: , + }, + { + path: routes.REQUIRED_INFORMATION, + element: , + }, + { + path: routes.EMAIL_VERIFICATION, + element: , + }, + { + path: routes.OPTIONAL_INFORMATION, + element: , }, { - path: "/connect-company", - element: , + path: "/success", + element: , }, { - path: "/profile-info", - element: , + path: "/confirm-identity", + element: , }, { path: routes.NOTIFICATIONS, diff --git a/src/pages/onboarding/ConfirmIdentity.tsx b/src/pages/onboarding/ConfirmIdentity.tsx new file mode 100644 index 0000000..192b3e5 --- /dev/null +++ b/src/pages/onboarding/ConfirmIdentity.tsx @@ -0,0 +1,186 @@ +import { useNavigate } from "react-router-dom"; +import { FormattedMessage } from "react-intl"; +import { Button } from "@/components/ui/button"; +import { Separator } from "@/components/ui/separator"; + +function RequiredInformation() { + return ( +
+
+ + + +
+ +
+
+ + + + John Doe +
+ + + +
+ + + + + example@bit.cr + +
+ + + +
+ + + + + The One Street 86, 11490, Nowhere + +
+
+
+ ); +} + +function OptionalInformation() { + return ( +
+
+ + + +
+ +
+
+ + + + - +
+ + + +
+ + + + - +
+ + + +
+ + + + - +
+ + + +
+ + + + - +
+ +
+ + + + - +
+
+
+ ); +} + +export default function ConfirmIdentity() { + const navigate = useNavigate(); + + const signToConfirm = () => navigate("/success"); + + return ( +
+
+

+ +

+ + + + +
+ +
+ + +
+ + +
+ ); +} diff --git a/src/pages/onboarding/CreateIdentity.tsx b/src/pages/onboarding/CreateIdentity.tsx deleted file mode 100644 index 581cfac..0000000 --- a/src/pages/onboarding/CreateIdentity.tsx +++ /dev/null @@ -1,367 +0,0 @@ -import { z } from "zod"; -import { zodResolver } from "@hookform/resolvers/zod"; -import { useForm } from "react-hook-form"; -import { - Select, - SelectContent, - SelectItem, - SelectTrigger, - SelectValue, -} from "@/components/ui/select"; -import { Input } from "@/components/ui/input"; -import { - Form, - FormControl, - FormField, - FormItem, - FormLabel, -} from "@/components/ui/form"; -import { Button } from "@/components/ui/button"; - -import defaultAvatar from "@/assets/default-avatar.svg"; -import { useState } from "react"; -import { Checkbox } from "@/components/ui/checkbox"; - -const createIdentityFormSchema = z.object({ - fullName: z.string().min(1, "Full name is required"), - dateOfBirth: z.object({ - day: z.string().regex(/^(0?[1-9]|[12][0-9]|3[01])$/, "Invalid day"), - month: z.string().regex(/^(0?[1-9]|1[0-2])$/, "Invalid month"), - year: z - .string() - .regex(/^\d{4}$/, "Invalid year") - .refine((val) => parseInt(val) <= new Date().getFullYear()), - }), - city: z.string().min(1), - country: z.string().min(1), - address: z.string().min(1), - email: z.string().email(), - taxId: z.string(), - - identityDocument: z - .instanceof(File) - .refine((file) => file.size < 5 * 1024 * 1024), -}); - -function CreateIdentityForm() { - const [isPreviewing, setIsPreviewing] = useState(false); - const [hasAgreedToTerms, setHasAgreedToTerms] = useState(false); - - const form = useForm>({ - resolver: zodResolver(createIdentityFormSchema), - defaultValues: { - fullName: "", - dateOfBirth: { - day: "", - month: "", - year: "", - }, - city: "", - country: "", - address: "", - email: "", - taxId: "", - }, - }); - - function onSubmit(values: z.infer) { - console.log(values); - - setIsPreviewing(true); - } - - return ( -
- -
- default-avatar -
- {isPreviewing ? "" : "Add photo"} -
-
- ( - - Full name - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - -
- {isPreviewing ? ( -
-
Date of birth
-
- {form.watch("dateOfBirth.day")}- - {form.watch("dateOfBirth.month")}- - {form.watch("dateOfBirth.year")} -
-
- ) : ( - <> - ( - - Date - - - - - )} - /> - - ( - - - - - - )} - /> - - ( - - - - - - )} - /> - - )} -
- - ( - - - City of birth - - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - - ( - - - Country of birth - - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - - ( - - - Postal address - - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - - ( - - Email - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - - ( - - - Social security number - - - {isPreviewing ? ( -
- {field.value} -
- ) : ( - - )} -
-
- )} - /> - - ( - - Identity document - - {isPreviewing ? ( -
- {value.name} -
- ) : ( - - onChange(event.target.files && event.target.files[0]) - } - /> - )} -
-
- )} - /> - - {isPreviewing ? ( -
-
- setHasAgreedToTerms(!hasAgreedToTerms)} - /> - -
-
- - -
-
- ) : ( - - )} - - - ); -} - -function CreateIdentity() { - return ( -
-

Create Identity

-
- -
-
- ); -} - -export default CreateIdentity; diff --git a/src/pages/onboarding/CreateNewIdentity.tsx b/src/pages/onboarding/CreateNewIdentity.tsx new file mode 100644 index 0000000..cc17247 --- /dev/null +++ b/src/pages/onboarding/CreateNewIdentity.tsx @@ -0,0 +1,51 @@ +import { useNavigate } from "react-router-dom"; +import { FormattedMessage } from "react-intl"; +import { Button } from "@/components/ui/button"; +import encryptedDataIllustration from "@/assets/encrypted-data-illustration.svg"; + +export default function CreateNewIdentity() { + const navigate = useNavigate(); + + const startCreatingIdentity = () => navigate("/required-information"); + + return ( +
+
+
+

+ +

+ + + + +
+ + Encrypted data +
+ + +
+ ); +} diff --git a/src/pages/onboarding/EmailVerification.tsx b/src/pages/onboarding/EmailVerification.tsx new file mode 100644 index 0000000..ff11912 --- /dev/null +++ b/src/pages/onboarding/EmailVerification.tsx @@ -0,0 +1,121 @@ +import { useRef } from "react"; +import { useNavigate } from "react-router-dom"; +import { FormattedMessage } from "react-intl"; +import { ChevronRightIcon } from "lucide-react"; +import { Button } from "@/components/ui/button"; + +function CodeInputs() { + const inputRefs = useRef<(HTMLInputElement | null)[]>([]); + + const handleInput = ( + e: React.ChangeEvent, + index: number + ) => { + const value = e.target.value; + + if (!/^\d*$/.test(value)) { + e.target.value = ""; + return; + } + + if (value.length === 1 && index < inputRefs.current.length - 1) { + inputRefs.current[index + 1]?.focus(); + } + }; + + const handleKeyDown = ( + e: React.KeyboardEvent, + index: number + ) => { + if (e.key === "Backspace" && e.currentTarget.value === "" && index > 0) { + inputRefs.current[index - 1]?.focus(); + } + }; + + return ( +
+ {[...Array(5)].map((_, index) => ( + (inputRefs.current[index] = el)} + className="w-12 h-12 border border-[#1B0F0014] rounded-[8px] bg-elevation-200 text-text-300 text-3xl font-light text-center focus:outline-none" + style={{ fontSize: "28px" }} + onChange={(e) => handleInput(e, index)} + onKeyDown={(e) => handleKeyDown(e, index)} + /> + ))} +
+ ); +} + +export default function EmailVerification() { + const navigate = useNavigate(); + + const goToOptionalInformation = () => navigate("/optional-information"); + + return ( +
+
+
+

+ +

+ + + + +
+ +
+ + + + + +
+
+ + +
+ ); +} diff --git a/src/pages/onboarding/OptionalInformation.tsx b/src/pages/onboarding/OptionalInformation.tsx new file mode 100644 index 0000000..40e09a8 --- /dev/null +++ b/src/pages/onboarding/OptionalInformation.tsx @@ -0,0 +1,103 @@ +import { useNavigate } from "react-router-dom"; +import { FormattedMessage } from "react-intl"; +import { Button } from "@/components/ui/button"; +import { Input } from "@/components/ui/input"; +import { DocumentUpload } from "@/components/DocumentUpload"; +import { + Select, + SelectGroup, + SelectValue, + SelectTrigger, + SelectContent, + SelectItem, +} from "@/components/ui/select"; + +export default function OptionalInformation() { + const navigate = useNavigate(); + + const confirmIdentity = () => navigate("/confirm-identity"); + + return ( +
+
+
+

+ +

+ + + + +
+ +
+ + + + +
+
+ +
+ + + +
+
+ ); +} diff --git a/src/pages/onboarding/RequiredInformation.tsx b/src/pages/onboarding/RequiredInformation.tsx new file mode 100644 index 0000000..fee2ef6 --- /dev/null +++ b/src/pages/onboarding/RequiredInformation.tsx @@ -0,0 +1,51 @@ +import { useNavigate } from "react-router-dom"; +import { FormattedMessage } from "react-intl"; +import { Button } from "@/components/ui/button"; +import { Input } from "@/components/ui/input"; + +export default function RequiredInformation() { + const navigate = useNavigate(); + + const verifyEmail = () => navigate("/email-verification"); + + return ( +
+
+
+

+ +

+ + + + +
+ +
+ + + +
+
+ + +
+ ); +} diff --git a/src/pages/onboarding/Success.tsx b/src/pages/onboarding/Success.tsx new file mode 100644 index 0000000..bd761d2 --- /dev/null +++ b/src/pages/onboarding/Success.tsx @@ -0,0 +1,43 @@ +import { FormattedMessage } from "react-intl"; +import { Button } from "@/components/ui/button"; +import successIllustration from "@/assets/success-illustration.svg"; + +export default function Success() { + return ( +
+
+
+

+ +

+ + + + +
+ + Success +
+ + +
+ ); +} diff --git a/src/utils/index.ts b/src/utils/index.ts index 99ff25b..0497682 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -11,3 +11,28 @@ export const detectBrowserLanguage = () => { return language; }; + +export const formatFileSize = (size: number) => { + if (size < 1024) return `${size} B`; + + if (size < 1048576) return `${(size / 1024).toFixed(2)} KB`; + + return `${(size / 1048576).toFixed(2)} MB`; +}; + +export const truncateFileName = (name: string) => { + const maxNameLength = 20; + const extensionIndex = name.lastIndexOf("."); + + if (extensionIndex === -1) + return name.length > maxNameLength + ? `${name.slice(0, 5)}...${name.slice(-3)}` + : name; + + const extension = name.slice(extensionIndex); + const baseName = name.slice(0, extensionIndex); + + return baseName.length > maxNameLength + ? `${baseName.slice(0, 5)}...${baseName.slice(-3)}${extension}` + : name; +};