Files
LetterFeed/frontend/components/letterfeed/EditNewsletterDialog.tsx
2025-07-17 14:36:53 +02:00

187 lines
6.1 KiB
TypeScript

import { useState, useEffect } from "react"
import { Button } from "@/components/ui/button"
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
} from "@/components/ui/dialog"
import { Input } from "@/components/ui/input"
import { Label } from "@/components/ui/label"
import { Plus } from "lucide-react"
import { Newsletter, updateNewsletter, deleteNewsletter } from "@/lib/api"
import { Checkbox } from "@/components/ui/checkbox"
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select"
interface EditNewsletterDialogProps {
newsletter: Newsletter | null
isOpen: boolean
folderOptions: string[]
onOpenChange: (isOpen: boolean) => void
onSuccess: () => void
}
export function EditNewsletterDialog({ newsletter, isOpen, folderOptions, onOpenChange, onSuccess }: EditNewsletterDialogProps) {
const [editedDetails, setEditedDetails] = useState<{ name: string; emails: string[], move_to_folder: string | null, extract_content: boolean }>({
name: "",
emails: [],
move_to_folder: "",
extract_content: false,
})
useEffect(() => {
if (newsletter) {
setEditedDetails({
name: newsletter.name,
emails: newsletter.senders.map((s) => s.email),
move_to_folder: newsletter.move_to_folder || "",
extract_content: newsletter.extract_content,
})
}
}, [newsletter])
if (!newsletter) return null
const handleUpdateEmailChange = (index: number, value: string) => {
setEditedDetails((prev) => ({
...prev,
emails: prev.emails.map((email, i) => (i === index ? value : email)),
}))
}
const handleAddEmailToEdit = () => {
setEditedDetails((prev) => ({
...prev,
emails: [...prev.emails, ""],
}))
}
const handleRemoveEmailFromEdit = (index: number) => {
setEditedDetails((prev) => ({
...prev,
emails: prev.emails.filter((_, i) => i !== index),
}))
}
const handleUpdate = async () => {
try {
await updateNewsletter(newsletter.id, {
name: editedDetails.name,
sender_emails: editedDetails.emails.filter((email) => email.trim()),
move_to_folder: editedDetails.move_to_folder,
extract_content: editedDetails.extract_content,
})
onOpenChange(false)
onSuccess()
} catch (error) {
console.error("Failed to update newsletter:", error)
}
}
const handleDelete = async () => {
if (window.confirm(`Are you sure you want to delete the "${newsletter.name}" newsletter?`)) {
try {
await deleteNewsletter(newsletter.id)
onOpenChange(false)
onSuccess()
} catch (error) {
console.error("Failed to delete newsletter:", error)
}
}
}
return (
<Dialog open={isOpen} onOpenChange={onOpenChange}>
<DialogContent className="sm:max-w-[425px]">
<DialogHeader>
<DialogTitle>Edit Newsletter</DialogTitle>
<DialogDescription>Update the details for {newsletter.name}.</DialogDescription>
</DialogHeader>
<div className="grid gap-4 py-4">
<div className="space-y-2">
<Label htmlFor="edit-name">Newsletter Name</Label>
<Input
id="edit-name"
value={editedDetails.name}
onChange={(e) => setEditedDetails((prev) => ({ ...prev, name: e.target.value }))}
/>
</div>
<div className="space-y-2">
<Label htmlFor="edit-move_to_folder">Move To Folder</Label>
<Select
value={editedDetails.move_to_folder || "None"}
onValueChange={(value) =>
setEditedDetails((prev) => ({ ...prev, move_to_folder: value === "None" ? "" : value }))
}
>
<SelectTrigger>
<SelectValue placeholder="Select folder or leave empty" />
</SelectTrigger>
<SelectContent>
<SelectItem value="None">Default (use global setting)</SelectItem>
{folderOptions.map((folder) => (
<SelectItem key={folder} value={folder}>
{folder}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="space-y-2">
<Label>Email Addresses</Label>
{editedDetails.emails.map((email, index) => (
<div key={index} className="flex gap-2">
<Input
value={email}
onChange={(e) => handleUpdateEmailChange(index, e.target.value)}
placeholder="Enter email address"
type="email"
/>
{editedDetails.emails.length > 1 && (
<Button variant="outline" size="sm" onClick={() => handleRemoveEmailFromEdit(index)}>
Remove
</Button>
)}
</div>
))}
<Button variant="outline" size="sm" onClick={handleAddEmailToEdit} className="w-full bg-transparent">
<Plus className="w-4 h-4 mr-2" />
Add Another Email
</Button>
</div>
<div className="flex items-center space-x-2">
<Checkbox
id="edit-extract-content"
checked={editedDetails.extract_content}
onCheckedChange={(checked) =>
setEditedDetails((prev) => ({ ...prev, extract_content: !!checked }))
}
/>
<Label htmlFor="edit-extract-content">Extract main content from emails</Label>
</div>
</div>
<DialogFooter className="sm:justify-between">
<Button variant="destructive" onClick={handleDelete}>
Delete Newsletter
</Button>
<div className="flex gap-2">
<Button variant="outline" onClick={() => onOpenChange(false)}>
Cancel
</Button>
<Button onClick={handleUpdate}>Save Changes</Button>
</div>
</DialogFooter>
</DialogContent>
</Dialog>
)
}