refactor: reorganize directory structure by moving UI components into feature-specific folders

This commit is contained in:
ramvignesh-b
2026-04-24 17:01:35 +05:30
parent c562c99d3a
commit db31be4ec8
11 changed files with 74 additions and 74 deletions
+3 -3
View File
@@ -1,10 +1,10 @@
import { FeatherIcon } from "@phosphor-icons/react"; import { FeatherIcon } from "@phosphor-icons/react";
import { useState } from "react"; import { useState } from "react";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { DrawerSection } from "../components/drawer/DrawerSection.tsx";
import { LetterItem } from "../components/drawer/LetterItem.tsx";
import { PasskeyModal } from "../components/drawer/PasskeyModal.tsx";
import Logo from "../components/Logo"; import Logo from "../components/Logo";
import { DrawerSection } from "../components/ui/DrawerSection";
import { LetterItem } from "../components/ui/LetterItem";
import { PasskeyModal } from "../components/ui/PasskeyModal";
import { PATHS } from "../config/routes"; import { PATHS } from "../config/routes";
import { useAuth } from "../hooks/useAuth"; import { useAuth } from "../hooks/useAuth";
import { useLetters } from "../hooks/useLetters"; import { useLetters } from "../hooks/useLetters";
+11 -11
View File
@@ -13,7 +13,7 @@ import Editor from "./Editor";
const API_URL = import.meta.env.VITE_API_URL; const API_URL = import.meta.env.VITE_API_URL;
// Mock ComposeCanvas to avoid Fabric.js issues and check readOnly prop // Mock ComposeCanvas to avoid Fabric.js issues and check readOnly prop
vi.mock("../components/ui/ComposeCanvas", () => ({ vi.mock("../components/editor/ComposeCanvas", () => ({
ComposeCanvas: vi.fn(({ readOnly }) => ( ComposeCanvas: vi.fn(({ readOnly }) => (
<div data-testid="canvas" data-readonly={readOnly}> <div data-testid="canvas" data-readonly={readOnly}>
Canvas Canvas
@@ -24,21 +24,21 @@ vi.mock("../components/ui/ComposeCanvas", () => ({
// Mock CryptoUtils to avoid real crypto calls in UI tests // Mock CryptoUtils to avoid real crypto calls in UI tests
vi.mock("../utils/crypto", () => { vi.mock("../utils/crypto", () => {
return { return {
CryptoUtils: () => ({ CryptoUtils: class {
initialize: vi.fn().mockResolvedValue(undefined), initialize = vi.fn().mockResolvedValue(undefined);
encryptLetter: vi.fn().mockResolvedValue({ encryptLetter = vi.fn().mockResolvedValue({
encrypted_content: "enc-content", encrypted_content: "enc-content",
encrypted_dek: "enc-dek", encrypted_dek: "enc-dek",
sharingKey: "share-key", sharingKey: "share-key",
}), });
encryptMetadata: vi.fn().mockResolvedValue({ encryptMetadata = vi.fn().mockResolvedValue({
encrypted_content: "enc-meta", encrypted_content: "enc-meta",
encrypted_dek: "enc-dek", encrypted_dek: "enc-dek",
}), });
decryptMetadata: vi.fn().mockResolvedValue({ recipient: "Test User" }), decryptMetadata = vi.fn().mockResolvedValue({ recipient: "Test User" });
decryptLetter: vi.fn().mockResolvedValue("{}"), decryptLetter = vi.fn().mockResolvedValue("{}");
extractSharingKey: vi.fn().mockResolvedValue("share-key"), extractSharingKey = vi.fn().mockResolvedValue("share-key");
}), },
}; };
}); });
+4 -4
View File
@@ -11,16 +11,16 @@ import {
useParams, useParams,
} from "react-router-dom"; } from "react-router-dom";
import { api } from "../api/apiClient"; import { api } from "../api/apiClient";
import {
type CanvasTools,
ComposeCanvas,
} from "../components/editor/ComposeCanvas";
import { PostSealModal } from "../components/editor/PostSealModal"; import { PostSealModal } from "../components/editor/PostSealModal";
import { import {
LetterHead, LetterHead,
ToolBar, ToolBar,
VaultConfirmModal, VaultConfirmModal,
} from "../components/editor/ToolBar"; } from "../components/editor/ToolBar";
import {
type CanvasTools,
ComposeCanvas,
} from "../components/ui/ComposeCanvas";
import DateDisplay from "../components/ui/DateDisplay"; import DateDisplay from "../components/ui/DateDisplay";
import { LogModal } from "../components/ui/LogModal"; import { LogModal } from "../components/ui/LogModal";
import { Navbar } from "../components/ui/Navbar"; import { Navbar } from "../components/ui/Navbar";
+52 -52
View File
@@ -20,6 +20,57 @@ const loginSchema = z.object({
type LoginInputs = z.infer<typeof loginSchema>; type LoginInputs = z.infer<typeof loginSchema>;
function WelcomeModal({ setShowWelcome }) {
return (
<div className="modal modal-open backdrop-blur-sm transition-all duration-1000">
<div className="modal-box border border-primary/20 shadow-2xl p-8">
<div className="flex flex-col items-center text-center gap-4">
<div className="bg-primary/10 p-4 rounded-full animate-pulse">
<ShieldCheckIcon
size={48}
weight="duotone"
className="text-primary"
/>
</div>
<h3 className="font-display text-2xl font-bold text-primary">
Welcome to <Logo />!
</h3>
<p className="text-base-content/80 leading-relaxed">
To ensure <span className="font-bold">complete privacy</span>, all
your letters are{" "}
<span className="font-bold underline">
sealed with your password
</span>
, which only you have access to.
<br />
<span className="font-bold">
The server never sees it, and it's a solemn promise!
</span>
</p>
<div className="alert alert-warning bg-paper/20 border-paper/20 flex items-start gap-3 text-left py-3">
<WarningIcon size={24} weight="fill" className="shrink-0 mt-0.5" />
<p className="text-sm font-medium text-primary-content">
If you ever happen to forget your password, your letters are lost
to time, forever.
</p>
</div>
<div className="modal-action w-full">
<button
type="button"
onClick={() => setShowWelcome(false)}
className="btn btn-primary w-full shadow-lg"
>
I understand
</button>
</div>
</div>
</div>
</div>
);
}
export default function Login() { export default function Login() {
const navigate = useNavigate(); const navigate = useNavigate();
const location = useLocation(); const location = useLocation();
@@ -74,58 +125,7 @@ export default function Login() {
return ( return (
<div className="flex flex-col gap-3"> <div className="flex flex-col gap-3">
{showWelcome && ( {showWelcome && <WelcomeModal setShowWelcome={setShowWelcome} />}
<div className="modal modal-open backdrop-blur-sm transition-all duration-1000">
<div className="modal-box border border-primary/20 shadow-2xl p-8">
<div className="flex flex-col items-center text-center gap-4">
<div className="bg-primary/10 p-4 rounded-full animate-pulse">
<ShieldCheckIcon
size={48}
weight="duotone"
className="text-primary"
/>
</div>
<h3 className="font-display text-2xl font-bold text-primary">
Welcome to <Logo />!
</h3>
<p className="text-base-content/80 leading-relaxed">
To ensure <span className="font-bold">complete privacy</span>,
all your letters are{" "}
<span className="font-bold underline">
sealed with your password
</span>
, which only you have access to.
<br />
<span className="font-bold">
The server never sees it, and it's a solemn promise!
</span>
</p>
<div className="alert alert-warning bg-paper/20 border-paper/20 flex items-start gap-3 text-left py-3">
<WarningIcon
size={24}
weight="fill"
className="shrink-0 mt-0.5"
/>
<p className="text-sm font-medium text-primary-content">
If you ever happen to forget your password, your letters are
lost to time, forever.
</p>
</div>
<div className="modal-action w-full">
<button
type="button"
onClick={() => setShowWelcome(false)}
className="btn btn-primary w-full shadow-lg"
>
I understand
</button>
</div>
</div>
</div>
</div>
)}
<div className="glass-card w-full max-w-sm p-2 transition-all duration-500 hover:shadow-2xl fade-zoom"> <div className="glass-card w-full max-w-sm p-2 transition-all duration-500 hover:shadow-2xl fade-zoom">
<form onSubmit={handleSubmit(onSubmit)} className="card-body gap-4"> <form onSubmit={handleSubmit(onSubmit)} className="card-body gap-4">
<h1 className="card-title font-display text-2xl font-bold justify-center text-primary tracking-tight"> <h1 className="card-title font-display text-2xl font-bold justify-center text-primary tracking-tight">
+3 -3
View File
@@ -8,13 +8,13 @@ import {
import { useEffect, useRef, useState } from "react"; import { useEffect, useRef, useState } from "react";
import { useLocation, useNavigate, useParams } from "react-router-dom"; import { useLocation, useNavigate, useParams } from "react-router-dom";
import { api } from "../api/apiClient"; import { api } from "../api/apiClient";
import Logo from "../components/Logo";
import { import {
type CanvasJSON, type CanvasJSON,
type CanvasTools, type CanvasTools,
ComposeCanvas, ComposeCanvas,
} from "../components/ui/ComposeCanvas"; } from "../components/editor/ComposeCanvas";
import { EnvelopeReveal } from "../components/ui/EnvelopeReveal"; import Logo from "../components/Logo";
import { EnvelopeReveal } from "../components/reader/EnvelopeReveal";
import { LogModal } from "../components/ui/LogModal"; import { LogModal } from "../components/ui/LogModal";
import { endpoints } from "../config/endpoints"; import { endpoints } from "../config/endpoints";
import { PATHS, ROUTES } from "../config/routes"; import { PATHS, ROUTES } from "../config/routes";
+1 -1
View File
@@ -2,7 +2,7 @@ import { api } from "../api/apiClient";
import type { import type {
CanvasJSON, CanvasJSON,
FabricImageJSON, FabricImageJSON,
} from "../components/ui/ComposeCanvas"; } from "../components/editor/ComposeCanvas";
import type { CryptoUtils } from "./crypto"; import type { CryptoUtils } from "./crypto";
import { blobUrlToFile } from "./fileUtils"; import { blobUrlToFile } from "./fileUtils";