mirror of
https://github.com/Mail-0/Zero.git
synced 2026-07-01 08:16:28 +00:00
feat: add demo mail layout with sample email data
- Created demo JSON with sample email threads - Implemented DemoMailLayout component for static mail preview - Added demo components for MailList and ThreadDisplay - Updated home page to include demo mail layout - Introduced conditional rendering for demo mode in mail components
This commit is contained in:
@@ -3,6 +3,7 @@
|
||||
import HeroImage from "@/components/home/hero-image";
|
||||
import Navbar from "@/components/home/navbar";
|
||||
import Hero from "@/components/home/hero";
|
||||
import { DemoMailLayout } from "@/components/mail/mail";
|
||||
|
||||
export default function Home() {
|
||||
return (
|
||||
@@ -13,7 +14,9 @@ export default function Home() {
|
||||
<div className="relative mx-auto mb-4 flex flex-col">
|
||||
<Navbar />
|
||||
<Hero />
|
||||
<HeroImage />
|
||||
<div className="container mx-auto">
|
||||
<DemoMailLayout />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
111
apps/mail/components/mail/demo.json
Normal file
111
apps/mail/components/mail/demo.json
Normal file
@@ -0,0 +1,111 @@
|
||||
[
|
||||
{
|
||||
"id": "love-001",
|
||||
"threadId": "love-001",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"personal"
|
||||
],
|
||||
"title": "🚨 EMERGENCY: I Might Be in Love with You 🚨",
|
||||
"body": "Hey... so, I think I caught feelings. Is there a cure, or am I doomed? 😳",
|
||||
"receivedOn": "2024-02-14",
|
||||
"sender": {
|
||||
"name": "Secret Admirer",
|
||||
"email": "mystery@unknown.com"
|
||||
},
|
||||
"unread": true,
|
||||
"subject": "HELP: Symptoms Include Butterflies and Overthinking",
|
||||
"totalReplies": 3,
|
||||
"decodedBody": "<p>Hey... so, I think I caught feelings. Is there a cure, or am I doomed? 😳</p>"
|
||||
},
|
||||
{
|
||||
"id": "job-003",
|
||||
"threadId": "job-003",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"work"
|
||||
],
|
||||
"title": "⚡ URGENT: Someone Wants to Pay You to Exist",
|
||||
"body": "Okay, not quite, but we do have a job for you. Are you in?",
|
||||
"receivedOn": "2024-03-05",
|
||||
"sender": {
|
||||
"name": "Tech Recruiter",
|
||||
"email": "hr@desperateforhelp.com"
|
||||
},
|
||||
"unread": false,
|
||||
"subject": "Please Work for Us, We Are Begging",
|
||||
"totalReplies": 4
|
||||
},
|
||||
{
|
||||
"id": "job-001",
|
||||
"threadId": "job-001",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"work"
|
||||
],
|
||||
"title": "🎉 You Got a Job! (Just Kidding, But Let’s Talk)",
|
||||
"body": "Hey, you look like someone who needs a paycheck. Want a job?",
|
||||
"receivedOn": "2024-03-01",
|
||||
"sender": {
|
||||
"name": "HR Recruiter",
|
||||
"email": "hireme@company.com"
|
||||
},
|
||||
"unread": false,
|
||||
"subject": "Work 9-5, Make Money, Repeat",
|
||||
"totalReplies": 5
|
||||
},
|
||||
{
|
||||
"id": "love-002",
|
||||
"threadId": "love-002",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"personal"
|
||||
],
|
||||
"title": "💔 I Wrote You a Poem (And It’s Terrible)",
|
||||
"body": "Roses are red, violets are blue, coffee this weekend, or should I be sad forever? ☕",
|
||||
"receivedOn": "2024-02-10",
|
||||
"sender": {
|
||||
"name": "Emma",
|
||||
"email": "emma@romantic.com"
|
||||
},
|
||||
"unread": true,
|
||||
"subject": "Worst Poem Ever, But With Love",
|
||||
"totalReplies": 2
|
||||
},
|
||||
{
|
||||
"id": "job-002",
|
||||
"threadId": "job-002",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"work"
|
||||
],
|
||||
"title": "💰 Work From Home & Become a Millionaire*",
|
||||
"body": "*Okay, maybe not a millionaire, but you’ll at least afford coffee. Interested?",
|
||||
"receivedOn": "2024-03-04",
|
||||
"sender": {
|
||||
"name": "John Doe",
|
||||
"email": "john.doe@scamfreejobs.com"
|
||||
},
|
||||
"unread": false,
|
||||
"subject": "No Boss, No Office, Just You & A Laptop",
|
||||
"totalReplies": 1
|
||||
},
|
||||
{
|
||||
"id": "love-003",
|
||||
"threadId": "love-003",
|
||||
"tags": [
|
||||
"inbox",
|
||||
"personal"
|
||||
],
|
||||
"title": "📅 Our First Date Was a Simulation (Or Was It?)",
|
||||
"body": "I had an amazing time! Unless it was all a dream. Let’s do it again to confirm.",
|
||||
"receivedOn": "2024-02-20",
|
||||
"sender": {
|
||||
"name": "Lily",
|
||||
"email": "lily@maybearealperson.com"
|
||||
},
|
||||
"unread": true,
|
||||
"subject": "Glitch in the Matrix? Or Just a Great Night?",
|
||||
"totalReplies": 1
|
||||
}
|
||||
]
|
||||
@@ -3,7 +3,7 @@
|
||||
import { Tooltip, TooltipContent, TooltipTrigger } from "@/components/ui/tooltip";
|
||||
import { ComponentProps, useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||
import { EmptyState, type FolderType } from "@/components/mail/empty-state";
|
||||
import { preloadThread, useThread, useThreads } from "@/hooks/use-threads";
|
||||
import { preloadThread, useThreads } from "@/hooks/use-threads";
|
||||
import { useSearchValue } from "@/hooks/use-search-value";
|
||||
import { markAsRead, markAsUnread } from "@/actions/mail";
|
||||
import { ScrollArea } from "@/components/ui/scroll-area";
|
||||
@@ -20,6 +20,7 @@ import { toast } from "sonner";
|
||||
import { ThreadContextMenu } from "../context/thread-context";
|
||||
import { useParams } from "next/navigation";
|
||||
import { useSummary } from "@/hooks/use-summary";
|
||||
import items from './demo.json'
|
||||
|
||||
interface MailListProps {
|
||||
isCompact?: boolean;
|
||||
@@ -34,6 +35,7 @@ type ThreadProps = {
|
||||
selectMode: MailSelectMode;
|
||||
onSelect: (message: InitialThread) => void;
|
||||
isCompact?: boolean;
|
||||
demo?: boolean;
|
||||
};
|
||||
|
||||
const highlightText = (text: string, highlight: string) => {
|
||||
@@ -56,7 +58,7 @@ const highlightText = (text: string, highlight: string) => {
|
||||
});
|
||||
};
|
||||
|
||||
const Thread = ({ message, selectMode, onSelect, isCompact }: ThreadProps) => {
|
||||
const Thread = ({ message, selectMode, onSelect, isCompact, demo }: ThreadProps) => {
|
||||
const { folder } = useParams<{ folder: string }>()
|
||||
const [mail] = useMail();
|
||||
const { data: session } = useSession();
|
||||
@@ -64,17 +66,17 @@ const Thread = ({ message, selectMode, onSelect, isCompact }: ThreadProps) => {
|
||||
const isHovering = useRef<boolean>(false);
|
||||
const hasPrefetched = useRef<boolean>(false);
|
||||
const [searchValue] = useSearchValue();
|
||||
const { mutate } = useThreads(folder, undefined, searchValue.value, 20);
|
||||
const { data } = useSummary(message.id)
|
||||
const { mutate } = demo ? { mutate: async () => { } } : useThreads(folder, undefined, searchValue.value, 20);
|
||||
|
||||
const isMailSelected = message.id === mail.selected;
|
||||
const isMailBulkSelected = mail.bulkSelected.includes(message.id);
|
||||
|
||||
const handleMailClick = async () => {
|
||||
if (demo) return;
|
||||
onSelect(message);
|
||||
if ((!selectMode || selectMode === 'single') && !isMailSelected && message.unread) {
|
||||
try {
|
||||
await markAsRead({ ids: [message.id] }).then(() => mutate()).catch(console.error);
|
||||
await markAsRead({ ids: [message.id] }).then(() => mutate() as any).catch(console.error);
|
||||
} catch (error) {
|
||||
console.error("Error marking message as read:", error);
|
||||
}
|
||||
@@ -82,6 +84,7 @@ const Thread = ({ message, selectMode, onSelect, isCompact }: ThreadProps) => {
|
||||
};
|
||||
|
||||
const handleMouseEnter = () => {
|
||||
if (demo) return;
|
||||
isHovering.current = true;
|
||||
|
||||
// Prefetch only in single select mode
|
||||
@@ -236,6 +239,37 @@ const StreamingText = ({ text }: { text: string }) => {
|
||||
);
|
||||
};
|
||||
|
||||
export function MailListDemo({ isCompact }: MailListProps) {
|
||||
return <ScrollArea
|
||||
className="h-full pb-2"
|
||||
type="scroll"
|
||||
>
|
||||
<div
|
||||
className={cn(
|
||||
"relative min-h-[calc(100vh-4rem)] w-full",
|
||||
|
||||
)}
|
||||
>
|
||||
<div
|
||||
className="absolute left-0 top-0 w-full p-[8px]"
|
||||
>
|
||||
{items.map((item) => {
|
||||
return item ? (
|
||||
<Thread
|
||||
demo
|
||||
key={item.id}
|
||||
message={item}
|
||||
selectMode={'single'}
|
||||
onSelect={() => console.log('Selected')}
|
||||
isCompact={isCompact}
|
||||
/>
|
||||
) : null;
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
</ScrollArea>
|
||||
}
|
||||
|
||||
export function MailList({ isCompact }: MailListProps) {
|
||||
const { folder } = useParams<{ folder: string }>()
|
||||
const [mail, setMail] = useMail();
|
||||
|
||||
@@ -5,10 +5,10 @@ import { ResizableHandle, ResizablePanel, ResizablePanelGroup } from "@/componen
|
||||
import { Drawer, DrawerContent, DrawerHeader, DrawerTitle } from "@/components/ui/drawer";
|
||||
import { AlignVerticalSpaceAround, ArchiveX, BellOff, SearchIcon, X } from "lucide-react";
|
||||
import { useState, useCallback, useMemo, useEffect, ReactNode } from "react";
|
||||
import { ThreadDisplay } from "@/components/mail/thread-display";
|
||||
import { ThreadDisplay, ThreadDemo } from "@/components/mail/thread-display";
|
||||
import { useMediaQuery } from "../../hooks/use-media-query";
|
||||
import { useSearchValue } from "@/hooks/use-search-value";
|
||||
import { MailList } from "@/components/mail/mail-list";
|
||||
import { MailList, MailListDemo } from "@/components/mail/mail-list";
|
||||
import { useMail } from "@/components/mail/use-mail";
|
||||
import { SidebarToggle } from "../ui/sidebar-toggle";
|
||||
import { Skeleton } from "@/components/ui/skeleton";
|
||||
@@ -21,6 +21,114 @@ import { useSession } from "@/lib/auth-client";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { SearchBar } from "./search-bar";
|
||||
import { cn } from "@/lib/utils";
|
||||
import items from './demo.json'
|
||||
|
||||
export function DemoMailLayout() {
|
||||
const mail = {
|
||||
selected: 'demo'
|
||||
}
|
||||
const isMobile = false
|
||||
const isValidating = false
|
||||
const isLoading = false
|
||||
const isDesktop = true
|
||||
const [isCompact, setIsCompact] = useState(false)
|
||||
const [open, setOpen] = useState(false)
|
||||
const handleClose = () => setOpen(false)
|
||||
|
||||
return <TooltipProvider delayDuration={0}>
|
||||
<div className="rounded-inherit flex">
|
||||
<ResizablePanelGroup
|
||||
direction="horizontal"
|
||||
autoSaveId="mail-panel-layout"
|
||||
className="rounded-inherit gap-1.5 overflow-hidden"
|
||||
>
|
||||
<ResizablePanel
|
||||
className={cn(
|
||||
"border-none !bg-transparent",
|
||||
mail?.selected ? "md:hidden lg:block" : "", // Hide on md, but show again on lg and up
|
||||
)}
|
||||
defaultSize={isMobile ? 100 : 25}
|
||||
minSize={isMobile ? 100 : 25}
|
||||
>
|
||||
<div className="bg-offsetLight dark:bg-offsetDark flex-1 flex-col overflow-y-auto shadow-inner md:flex md:rounded-2xl md:border md:shadow-sm">
|
||||
<div className={cn("compose-gradient h-0.5 w-full transition-opacity", isValidating ? "opacity-50" : "opacity-0")} />
|
||||
<div
|
||||
className={cn(
|
||||
"sticky top-0 z-10 flex items-center justify-between gap-1.5 p-2 transition-colors",
|
||||
)}
|
||||
>
|
||||
<SidebarToggle className="h-fit px-2" />
|
||||
<Button
|
||||
variant="ghost"
|
||||
className="md:h-fit md:px-2"
|
||||
onClick={() => setIsCompact(!isCompact)}
|
||||
>
|
||||
<AlignVerticalSpaceAround />
|
||||
</Button>
|
||||
</div>
|
||||
<div className="h-[calc(100dvh-56px)] overflow-hidden pt-0 md:h-[calc(100dvh-(8px+8px+14px+44px))]">
|
||||
{isLoading ? (
|
||||
<div className="flex flex-col">
|
||||
{[...Array(8)].map((_, i) => (
|
||||
<div key={i} className="flex flex-col px-4 py-3">
|
||||
<div className="flex w-full items-center justify-between">
|
||||
<div className="flex items-center gap-2">
|
||||
<Skeleton className="h-4 w-24" />
|
||||
</div>
|
||||
<Skeleton className="h-3 w-12" />
|
||||
</div>
|
||||
<Skeleton className="mt-2 h-3 w-32" />
|
||||
<Skeleton className="mt-2 h-3 w-full" />
|
||||
<div className="mt-2 flex gap-2">
|
||||
<Skeleton className="h-4 w-16 rounded-full" />
|
||||
<Skeleton className="h-4 w-16 rounded-full" />
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
) : (
|
||||
<MailListDemo
|
||||
isCompact={isCompact}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</ResizablePanel>
|
||||
|
||||
{isDesktop && mail.selected && (
|
||||
<>
|
||||
<ResizableHandle className="opacity-0" />
|
||||
<ResizablePanel
|
||||
className="shadow-sm md:flex md:rounded-2xl md:border md:shadow-sm bg-offsetLight dark:bg-offsetDark"
|
||||
defaultSize={75}
|
||||
minSize={25}
|
||||
>
|
||||
<div className="hidden h-[calc(100vh-(12px+14px))] flex-1 md:block relative">
|
||||
<ThreadDemo mail={[items[0]]} onClose={handleClose} />
|
||||
</div>
|
||||
</ResizablePanel>
|
||||
</>
|
||||
)}
|
||||
</ResizablePanelGroup>
|
||||
|
||||
{/* Mobile Drawer */}
|
||||
{isMobile && (
|
||||
<Drawer open={open} onOpenChange={setOpen}>
|
||||
<DrawerContent className="bg-offsetLight dark:bg-offsetDark h-[calc(100vh-3rem)] overflow-hidden p-0">
|
||||
<DrawerHeader className="sr-only">
|
||||
<DrawerTitle>Email Details</DrawerTitle>
|
||||
</DrawerHeader>
|
||||
<div className="flex h-full flex-col overflow-hidden">
|
||||
<div className="flex-1 overflow-hidden">
|
||||
<ThreadDisplay mail={mail.selected} onClose={handleClose} isMobile={true} />
|
||||
</div>
|
||||
</div>
|
||||
</DrawerContent>
|
||||
</Drawer>
|
||||
)}
|
||||
</div>
|
||||
</TooltipProvider>
|
||||
}
|
||||
|
||||
export function MailLayout() {
|
||||
const { folder } = useParams<{ folder: string }>()
|
||||
|
||||
@@ -27,11 +27,141 @@ import { cn } from "@/lib/utils";
|
||||
import React from "react";
|
||||
|
||||
interface ThreadDisplayProps {
|
||||
mail: string | null;
|
||||
mail: any;
|
||||
onClose?: () => void;
|
||||
isMobile?: boolean;
|
||||
}
|
||||
|
||||
export function ThreadDemo({ mail: emailData, onClose, isMobile }: ThreadDisplayProps) {
|
||||
const isFullscreen = false
|
||||
return <div
|
||||
className={cn(
|
||||
"flex flex-col",
|
||||
isFullscreen ? "h-screen" : isMobile ? "h-full" : "h-[calc(100vh-2rem)]",
|
||||
)}
|
||||
>
|
||||
<div
|
||||
className={cn(
|
||||
"bg-offsetLight dark:bg-offsetDark relative flex flex-col overflow-hidden transition-all duration-300",
|
||||
isMobile ? "h-full" : "h-full",
|
||||
!isMobile && !isFullscreen && "rounded-r-lg",
|
||||
isFullscreen ? "fixed inset-0 z-50" : "",
|
||||
)}
|
||||
>
|
||||
<div className="flex flex-shrink-0 items-center border-b p-2">
|
||||
<div className="flex flex-1 items-center gap-2">
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
<Button
|
||||
variant="ghost"
|
||||
className="md:h-fit md:px-2"
|
||||
disabled={!emailData}
|
||||
>
|
||||
<X className="h-4 w-4" />
|
||||
<span className="sr-only">Close</span>
|
||||
</Button>
|
||||
</TooltipTrigger>
|
||||
<TooltipContent>Close</TooltipContent>
|
||||
</Tooltip>
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
<Button
|
||||
variant="ghost"
|
||||
className="md:h-fit md:px-2"
|
||||
disabled={!emailData}
|
||||
>
|
||||
{isFullscreen ? (
|
||||
<Minimize2 className="h-4 w-4" />
|
||||
) : (
|
||||
<Maximize2 className="h-4 w-4" />
|
||||
)}
|
||||
<span className="sr-only">
|
||||
{isFullscreen ? "Exit fullscreen" : "Enter fullscreen"}
|
||||
</span>
|
||||
</Button>
|
||||
</TooltipTrigger>
|
||||
<TooltipContent>
|
||||
{isFullscreen ? "Exit fullscreen" : "Enter fullscreen"}
|
||||
</TooltipContent>
|
||||
</Tooltip>
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
<Button variant="ghost" className="md:h-fit md:px-2" disabled={!emailData}>
|
||||
<Archive className="h-4 w-4" />
|
||||
<span className="sr-only">Archive</span>
|
||||
</Button>
|
||||
</TooltipTrigger>
|
||||
<TooltipContent>Archive</TooltipContent>
|
||||
</Tooltip>
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
<Button
|
||||
variant="ghost"
|
||||
className="md:h-fit md:px-2"
|
||||
disabled={!emailData}
|
||||
>
|
||||
<Reply className="h-4 w-4" />
|
||||
<span className="sr-only">Reply</span>
|
||||
</Button>
|
||||
</TooltipTrigger>
|
||||
<TooltipContent>Reply</TooltipContent>
|
||||
</Tooltip>
|
||||
<DropdownMenu>
|
||||
<DropdownMenuTrigger asChild>
|
||||
<Button variant="ghost" className="md:h-fit md:px-2" disabled={!emailData}>
|
||||
<MoreVertical className="h-4 w-4" />
|
||||
<span className="sr-only">More</span>
|
||||
</Button>
|
||||
</DropdownMenuTrigger>
|
||||
<DropdownMenuContent align="end">
|
||||
<DropdownMenuItem>
|
||||
<ArchiveX className="mr-2 h-4 w-4" /> Move to spam
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem>
|
||||
<ReplyAll className="mr-2 h-4 w-4" /> Reply all
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem>
|
||||
<Forward className="mr-2 h-4 w-4" /> Forward
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem>Mark as unread</DropdownMenuItem>
|
||||
<DropdownMenuItem>Add label</DropdownMenuItem>
|
||||
<DropdownMenuItem>Mute thread</DropdownMenuItem>
|
||||
</DropdownMenuContent>
|
||||
</DropdownMenu>
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex min-h-0 flex-1 flex-col overflow-hidden">
|
||||
<ScrollArea className="flex-1" type="scroll">
|
||||
<div className="pb-4">
|
||||
{[...(emailData || [])].reverse().map((message, index) => (
|
||||
<div
|
||||
key={message.id}
|
||||
className={cn(
|
||||
"transition-all duration-200",
|
||||
index > 0 && "border-border border-t",
|
||||
)}
|
||||
>
|
||||
<MailDisplay
|
||||
emailData={message}
|
||||
isFullscreen={isFullscreen}
|
||||
isMuted={false}
|
||||
isLoading={false}
|
||||
index={index}
|
||||
/>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</ScrollArea>
|
||||
<div className="flex-shrink-0 relative top-1">
|
||||
<ReplyCompose emailData={emailData} isOpen={false} setIsOpen={() => { }} />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
|
||||
export function ThreadDisplay({ mail, onClose, isMobile }: ThreadDisplayProps) {
|
||||
const [, setMail] = useMail();
|
||||
const { data: emailData, isLoading } = useThread(mail ?? "");
|
||||
|
||||
@@ -2,6 +2,7 @@ import { EMAIL_HTML_TEMPLATE } from "./constants";
|
||||
import Color from "color";
|
||||
|
||||
export const template = (html: string) => {
|
||||
if (typeof DOMParser === "undefined") return html;
|
||||
const htmlParser = new DOMParser();
|
||||
const doc = htmlParser.parseFromString(html, "text/html");
|
||||
const template = htmlParser.parseFromString(EMAIL_HTML_TEMPLATE, "text/html");
|
||||
|
||||
Reference in New Issue
Block a user