mirror of
https://github.com/khoaliber/LetterFeed.git
synced 2026-03-02 13:18:27 +00:00
refactor: centralize newsletter dialogs and optimize newsletter card
This commit is contained in:
@@ -1,186 +0,0 @@
|
|||||||
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>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -3,7 +3,6 @@ import { Button } from "@/components/ui/button"
|
|||||||
import { Badge } from "@/components/ui/badge"
|
import { Badge } from "@/components/ui/badge"
|
||||||
import { Rss, Mail, ExternalLink, Edit } from "lucide-react"
|
import { Rss, Mail, ExternalLink, Edit } from "lucide-react"
|
||||||
import { Newsletter, getFeedUrl } from "@/lib/api"
|
import { Newsletter, getFeedUrl } from "@/lib/api"
|
||||||
import { useEffect, useState } from "react"
|
|
||||||
|
|
||||||
interface NewsletterCardProps {
|
interface NewsletterCardProps {
|
||||||
newsletter: Newsletter
|
newsletter: Newsletter
|
||||||
@@ -11,16 +10,7 @@ interface NewsletterCardProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function NewsletterCard({ newsletter, onEdit }: NewsletterCardProps) {
|
export function NewsletterCard({ newsletter, onEdit }: NewsletterCardProps) {
|
||||||
const [absoluteFeedUrl, setAbsoluteFeedUrl] = useState(getFeedUrl(newsletter.id))
|
const feedUrl = getFeedUrl(newsletter.id)
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const url = getFeedUrl(newsletter.id)
|
|
||||||
if (url.startsWith("/")) {
|
|
||||||
setAbsoluteFeedUrl(`${window.location.origin}${url}`)
|
|
||||||
} else {
|
|
||||||
setAbsoluteFeedUrl(url)
|
|
||||||
}
|
|
||||||
}, [newsletter.id])
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card className="hover:shadow-md transition-shadow flex flex-col">
|
<Card className="hover:shadow-md transition-shadow flex flex-col">
|
||||||
@@ -32,8 +22,8 @@ export function NewsletterCard({ newsletter, onEdit }: NewsletterCardProps) {
|
|||||||
{newsletter.name}
|
{newsletter.name}
|
||||||
</CardTitle>
|
</CardTitle>
|
||||||
<CardDescription>
|
<CardDescription>
|
||||||
{newsletter.entries_count} entr{newsletter.entries_count !== 1 ? "ies" : "y"}
|
{newsletter.entries_count} entr{newsletter.entries_count !== 1 ? "ies" : "y"}
|
||||||
</CardDescription>
|
</CardDescription>
|
||||||
</div>
|
</div>
|
||||||
<Button variant="ghost" size="icon" onClick={() => onEdit(newsletter)} aria-label="Edit Newsletter">
|
<Button variant="ghost" size="icon" onClick={() => onEdit(newsletter)} aria-label="Edit Newsletter">
|
||||||
<Edit className="w-4 h-4" />
|
<Edit className="w-4 h-4" />
|
||||||
@@ -58,13 +48,13 @@ export function NewsletterCard({ newsletter, onEdit }: NewsletterCardProps) {
|
|||||||
<div>
|
<div>
|
||||||
<h4 className="text-sm font-medium text-gray-700 mb-2">RSS Feed</h4>
|
<h4 className="text-sm font-medium text-gray-700 mb-2">RSS Feed</h4>
|
||||||
<a
|
<a
|
||||||
href={getFeedUrl(newsletter.id)}
|
href={feedUrl}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="inline-flex items-center gap-1 text-sm text-blue-600 hover:text-blue-800 hover:underline"
|
className="inline-flex items-center gap-1 text-sm text-blue-600 hover:text-blue-800 hover:underline"
|
||||||
>
|
>
|
||||||
<ExternalLink className="w-3 h-3" />
|
<ExternalLink className="w-3 h-3" />
|
||||||
{absoluteFeedUrl}
|
{feedUrl}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from "react"
|
import { useState, useEffect } from "react"
|
||||||
import { Button } from "@/components/ui/button"
|
import { Button } from "@/components/ui/button"
|
||||||
import {
|
import {
|
||||||
Dialog,
|
Dialog,
|
||||||
@@ -11,9 +11,8 @@ import {
|
|||||||
import { Input } from "@/components/ui/input"
|
import { Input } from "@/components/ui/input"
|
||||||
import { Label } from "@/components/ui/label"
|
import { Label } from "@/components/ui/label"
|
||||||
import { Plus } from "lucide-react"
|
import { Plus } from "lucide-react"
|
||||||
import { createNewsletter } from "@/lib/api"
|
import { Newsletter, createNewsletter, updateNewsletter, deleteNewsletter } from "@/lib/api"
|
||||||
import { Checkbox } from "@/components/ui/checkbox"
|
import { Checkbox } from "@/components/ui/checkbox"
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Select,
|
Select,
|
||||||
SelectContent,
|
SelectContent,
|
||||||
@@ -22,56 +21,95 @@ import {
|
|||||||
SelectValue,
|
SelectValue,
|
||||||
} from "@/components/ui/select"
|
} from "@/components/ui/select"
|
||||||
|
|
||||||
interface AddNewsletterDialogProps {
|
interface NewsletterDialogProps {
|
||||||
|
newsletter?: Newsletter | null
|
||||||
isOpen: boolean
|
isOpen: boolean
|
||||||
folderOptions: string[]
|
folderOptions: string[]
|
||||||
onOpenChange: (isOpen: boolean) => void
|
onOpenChange: (isOpen: boolean) => void
|
||||||
onSuccess: () => void
|
onSuccess: () => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuccess }: AddNewsletterDialogProps) {
|
const getInitialState = (newsletter: Newsletter | null | undefined) => {
|
||||||
const [newNewsletter, setNewNewsletter] = useState({
|
if (newsletter) {
|
||||||
|
return {
|
||||||
|
name: newsletter.name,
|
||||||
|
emails: newsletter.senders.map((s) => s.email),
|
||||||
|
move_to_folder: newsletter.move_to_folder || "",
|
||||||
|
extract_content: newsletter.extract_content,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
name: "",
|
name: "",
|
||||||
emails: [""],
|
emails: [""],
|
||||||
move_to_folder: "",
|
move_to_folder: "",
|
||||||
extract_content: false,
|
extract_content: false,
|
||||||
})
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function NewsletterDialog({ newsletter, isOpen, folderOptions, onOpenChange, onSuccess }: NewsletterDialogProps) {
|
||||||
|
const isEditMode = !!newsletter
|
||||||
|
const [formData, setFormData] = useState(getInitialState(newsletter))
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isOpen) {
|
||||||
|
setFormData(getInitialState(newsletter))
|
||||||
|
}
|
||||||
|
}, [isOpen, newsletter])
|
||||||
|
|
||||||
|
const handleEmailChange = (index: number, value: string) => {
|
||||||
|
setFormData((prev) => ({
|
||||||
|
...prev,
|
||||||
|
emails: prev.emails.map((email, i) => (i === index ? value : email)),
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
const handleAddEmail = () => {
|
const handleAddEmail = () => {
|
||||||
setNewNewsletter((prev) => ({
|
setFormData((prev) => ({
|
||||||
...prev,
|
...prev,
|
||||||
emails: [...prev.emails, ""],
|
emails: [...prev.emails, ""],
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleRemoveEmail = (index: number) => {
|
const handleRemoveEmail = (index: number) => {
|
||||||
setNewNewsletter((prev) => ({
|
setFormData((prev) => ({
|
||||||
...prev,
|
...prev,
|
||||||
emails: prev.emails.filter((_, i) => i !== index),
|
emails: prev.emails.filter((_, i) => i !== index),
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleEmailChange = (index: number, value: string) => {
|
const handleSubmit = async () => {
|
||||||
setNewNewsletter((prev) => ({
|
if (!formData.name || !formData.emails.some((email) => email.trim())) {
|
||||||
...prev,
|
return
|
||||||
emails: prev.emails.map((email, i) => (i === index ? value : email)),
|
}
|
||||||
}))
|
|
||||||
|
const payload = {
|
||||||
|
name: formData.name,
|
||||||
|
sender_emails: formData.emails.filter((email) => email.trim()),
|
||||||
|
move_to_folder: formData.move_to_folder,
|
||||||
|
extract_content: formData.extract_content,
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (isEditMode) {
|
||||||
|
await updateNewsletter(newsletter.id, payload)
|
||||||
|
} else {
|
||||||
|
await createNewsletter(payload)
|
||||||
|
}
|
||||||
|
onOpenChange(false)
|
||||||
|
onSuccess()
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Failed to ${isEditMode ? "update" : "create"} newsletter:`, error)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleSubmit = async () => {
|
const handleDelete = async () => {
|
||||||
if (newNewsletter.name && newNewsletter.emails.some((email) => email.trim())) {
|
if (isEditMode && window.confirm(`Are you sure you want to delete the "${newsletter.name}" newsletter?`)) {
|
||||||
try {
|
try {
|
||||||
await createNewsletter({
|
await deleteNewsletter(newsletter.id)
|
||||||
name: newNewsletter.name,
|
|
||||||
sender_emails: newNewsletter.emails.filter((email) => email.trim()),
|
|
||||||
move_to_folder: newNewsletter.move_to_folder,
|
|
||||||
extract_content: newNewsletter.extract_content,
|
|
||||||
})
|
|
||||||
setNewNewsletter({ name: "", emails: [""], move_to_folder: "", extract_content: false })
|
|
||||||
onOpenChange(false)
|
onOpenChange(false)
|
||||||
onSuccess()
|
onSuccess()
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Failed to create newsletter:", error)
|
console.error("Failed to delete newsletter:", error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -80,16 +118,18 @@ export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuc
|
|||||||
<Dialog open={isOpen} onOpenChange={onOpenChange}>
|
<Dialog open={isOpen} onOpenChange={onOpenChange}>
|
||||||
<DialogContent className="sm:max-w-[425px]">
|
<DialogContent className="sm:max-w-[425px]">
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle>Register New Newsletter</DialogTitle>
|
<DialogTitle>{isEditMode ? "Edit" : "Register New"} Newsletter</DialogTitle>
|
||||||
<DialogDescription>Add a new newsletter.</DialogDescription>
|
<DialogDescription>
|
||||||
|
{isEditMode ? `Update the details for ${newsletter.name}.` : "Add a new newsletter."}
|
||||||
|
</DialogDescription>
|
||||||
</DialogHeader>
|
</DialogHeader>
|
||||||
<div className="grid gap-4 py-4">
|
<div className="grid gap-4 py-4">
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
<Label htmlFor="name">Newsletter Name</Label>
|
<Label htmlFor="name">Newsletter Name</Label>
|
||||||
<Input
|
<Input
|
||||||
id="name"
|
id="name"
|
||||||
value={newNewsletter.name}
|
value={formData.name}
|
||||||
onChange={(e) => setNewNewsletter((prev) => ({ ...prev, name: e.target.value }))}
|
onChange={(e) => setFormData((prev) => ({ ...prev, name: e.target.value }))}
|
||||||
placeholder="Enter newsletter name"
|
placeholder="Enter newsletter name"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -97,9 +137,9 @@ export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuc
|
|||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
<Label htmlFor="move_to_folder">Move To Folder</Label>
|
<Label htmlFor="move_to_folder">Move To Folder</Label>
|
||||||
<Select
|
<Select
|
||||||
value={newNewsletter.move_to_folder || "None"}
|
value={formData.move_to_folder || "None"}
|
||||||
onValueChange={(value) =>
|
onValueChange={(value) =>
|
||||||
setNewNewsletter((prev) => ({ ...prev, move_to_folder: value === "None" ? "" : value }))
|
setFormData((prev) => ({ ...prev, move_to_folder: value === "None" ? "" : value }))
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<SelectTrigger>
|
<SelectTrigger>
|
||||||
@@ -118,7 +158,7 @@ export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuc
|
|||||||
|
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
<Label>Email Addresses</Label>
|
<Label>Email Addresses</Label>
|
||||||
{newNewsletter.emails.map((email, index) => (
|
{formData.emails.map((email, index) => (
|
||||||
<div key={index} className="flex gap-2">
|
<div key={index} className="flex gap-2">
|
||||||
<Input
|
<Input
|
||||||
value={email}
|
value={email}
|
||||||
@@ -126,7 +166,7 @@ export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuc
|
|||||||
placeholder="Enter email address"
|
placeholder="Enter email address"
|
||||||
type="email"
|
type="email"
|
||||||
/>
|
/>
|
||||||
{newNewsletter.emails.length > 1 && (
|
{formData.emails.length > 1 && (
|
||||||
<Button variant="outline" size="sm" onClick={() => handleRemoveEmail(index)}>
|
<Button variant="outline" size="sm" onClick={() => handleRemoveEmail(index)}>
|
||||||
Remove
|
Remove
|
||||||
</Button>
|
</Button>
|
||||||
@@ -141,19 +181,28 @@ export function AddNewsletterDialog({ isOpen, folderOptions, onOpenChange, onSuc
|
|||||||
<div className="flex items-center space-x-2">
|
<div className="flex items-center space-x-2">
|
||||||
<Checkbox
|
<Checkbox
|
||||||
id="extract-content"
|
id="extract-content"
|
||||||
checked={newNewsletter.extract_content}
|
checked={formData.extract_content}
|
||||||
onCheckedChange={(checked) =>
|
onCheckedChange={(checked) =>
|
||||||
setNewNewsletter((prev) => ({ ...prev, extract_content: !!checked }))
|
setFormData((prev) => ({ ...prev, extract_content: !!checked }))
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Label htmlFor="extract-content">Extract main content from emails</Label>
|
<Label htmlFor="extract-content">Extract main content from emails</Label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<DialogFooter>
|
<DialogFooter className={isEditMode ? "sm:justify-between" : ""}>
|
||||||
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
{isEditMode && (
|
||||||
Cancel
|
<Button variant="destructive" onClick={handleDelete}>
|
||||||
</Button>
|
Delete Newsletter
|
||||||
<Button onClick={handleSubmit}>Register Newsletter</Button>
|
</Button>
|
||||||
|
)}
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button onClick={handleSubmit}>
|
||||||
|
{isEditMode ? "Save Changes" : "Register Newsletter"}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</DialogFooter>
|
</DialogFooter>
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
import React from "react"
|
|
||||||
import { render, screen, fireEvent, waitFor } from "@testing-library/react"
|
|
||||||
import "@testing-library/jest-dom"
|
|
||||||
import { AddNewsletterDialog } from "../AddNewsletterDialog"
|
|
||||||
import * as api from "@/lib/api"
|
|
||||||
|
|
||||||
// Mock the API module
|
|
||||||
jest.mock("@/lib/api", () => ({
|
|
||||||
...jest.requireActual("@/lib/api"),
|
|
||||||
createNewsletter: jest.fn(),
|
|
||||||
}))
|
|
||||||
|
|
||||||
const mockedApi = api as jest.Mocked<typeof api>
|
|
||||||
|
|
||||||
describe("AddNewsletterDialog", () => {
|
|
||||||
const handleOpenChange = jest.fn()
|
|
||||||
const handleSuccess = jest.fn()
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks()
|
|
||||||
})
|
|
||||||
|
|
||||||
it("allows user to fill out the form and submit", async () => {
|
|
||||||
mockedApi.createNewsletter.mockResolvedValueOnce({
|
|
||||||
id: 1,
|
|
||||||
name: "My New Newsletter",
|
|
||||||
is_active: true,
|
|
||||||
extract_content: false,
|
|
||||||
senders: [{ id: 1, email: "test@example.com", newsletter_id: 1 }],
|
|
||||||
entries_count: 0,
|
|
||||||
})
|
|
||||||
|
|
||||||
render(<AddNewsletterDialog isOpen={true} folderOptions={["INBOX", "Archive"]} onOpenChange={handleOpenChange} onSuccess={handleSuccess} />)
|
|
||||||
|
|
||||||
// Fill out the form
|
|
||||||
fireEvent.change(screen.getByLabelText(/Newsletter Name/i), { target: { value: "My New Newsletter" } })
|
|
||||||
fireEvent.change(screen.getByPlaceholderText(/Enter email address/i), { target: { value: "test@example.com" } })
|
|
||||||
|
|
||||||
// Submit the form
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Register Newsletter/i }))
|
|
||||||
|
|
||||||
// Wait for the async operation to complete
|
|
||||||
await waitFor(() => {
|
|
||||||
expect(mockedApi.createNewsletter).toHaveBeenCalledWith({
|
|
||||||
name: "My New Newsletter",
|
|
||||||
sender_emails: ["test@example.com"],
|
|
||||||
move_to_folder: "",
|
|
||||||
extract_content: false,
|
|
||||||
})
|
|
||||||
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
|
||||||
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
it("allows adding and removing email fields", () => {
|
|
||||||
render(<AddNewsletterDialog isOpen={true} folderOptions={[]} onOpenChange={() => {}} onSuccess={() => {}} />)
|
|
||||||
|
|
||||||
// Initial state
|
|
||||||
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(1)
|
|
||||||
|
|
||||||
// Add another email
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Add Another Email/i }))
|
|
||||||
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(2)
|
|
||||||
|
|
||||||
// Remove the first email
|
|
||||||
fireEvent.click(screen.getAllByRole("button", { name: /Remove/i })[0])
|
|
||||||
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(1)
|
|
||||||
})
|
|
||||||
|
|
||||||
it("closes the dialog when cancel is clicked", () => {
|
|
||||||
render(<AddNewsletterDialog isOpen={true} folderOptions={["INBOX", "Archive"]} onOpenChange={handleOpenChange} onSuccess={handleSuccess} />)
|
|
||||||
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Cancel/i }))
|
|
||||||
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
|
||||||
expect(handleSuccess).not.toHaveBeenCalled()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
@@ -1,114 +0,0 @@
|
|||||||
import React from "react"
|
|
||||||
import { render, screen, fireEvent, waitFor } from "@testing-library/react"
|
|
||||||
import "@testing-library/jest-dom"
|
|
||||||
import { EditNewsletterDialog } from "../EditNewsletterDialog"
|
|
||||||
import { Newsletter } from "@/lib/api"
|
|
||||||
import * as api from "@/lib/api"
|
|
||||||
|
|
||||||
// Mock the API module
|
|
||||||
jest.mock("@/lib/api", () => ({
|
|
||||||
...jest.requireActual("@/lib/api"),
|
|
||||||
updateNewsletter: jest.fn(),
|
|
||||||
deleteNewsletter: jest.fn(),
|
|
||||||
}))
|
|
||||||
|
|
||||||
const mockedApi = api as jest.Mocked<typeof api>
|
|
||||||
|
|
||||||
const mockNewsletter: Newsletter = {
|
|
||||||
id: 1,
|
|
||||||
name: "Existing Newsletter",
|
|
||||||
is_active: true,
|
|
||||||
extract_content: false,
|
|
||||||
senders: [{ id: 1, email: "current@example.com", newsletter_id: 1 }],
|
|
||||||
entries_count: 5,
|
|
||||||
}
|
|
||||||
|
|
||||||
describe("EditNewsletterDialog", () => {
|
|
||||||
const handleOpenChange = jest.fn()
|
|
||||||
const handleSuccess = jest.fn()
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks()
|
|
||||||
// Mock window.confirm for the delete action
|
|
||||||
window.confirm = jest.fn(() => true)
|
|
||||||
})
|
|
||||||
|
|
||||||
it("renders with initial newsletter data and allows updates", async () => {
|
|
||||||
mockedApi.updateNewsletter.mockResolvedValueOnce({ ...mockNewsletter, name: "Updated Name" })
|
|
||||||
|
|
||||||
render(
|
|
||||||
<EditNewsletterDialog
|
|
||||||
newsletter={mockNewsletter}
|
|
||||||
isOpen={true}
|
|
||||||
folderOptions={["INBOX", "Archive"]}
|
|
||||||
onOpenChange={handleOpenChange}
|
|
||||||
onSuccess={handleSuccess}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
|
|
||||||
// Check that initial data is present
|
|
||||||
const nameInput = screen.getByLabelText(/Newsletter Name/i)
|
|
||||||
expect(nameInput).toHaveValue("Existing Newsletter")
|
|
||||||
expect(screen.getByDisplayValue("current@example.com")).toBeInTheDocument()
|
|
||||||
|
|
||||||
// Update the name
|
|
||||||
fireEvent.change(nameInput, { target: { value: "Updated Name" } })
|
|
||||||
|
|
||||||
// Submit
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Save Changes/i }))
|
|
||||||
|
|
||||||
await waitFor(() => {
|
|
||||||
expect(mockedApi.updateNewsletter).toHaveBeenCalledWith(1, {
|
|
||||||
name: "Updated Name",
|
|
||||||
sender_emails: ["current@example.com"],
|
|
||||||
move_to_folder: "",
|
|
||||||
extract_content: false,
|
|
||||||
})
|
|
||||||
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
|
||||||
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
it("calls deleteNewsletter when delete button is clicked and confirmed", async () => {
|
|
||||||
mockedApi.deleteNewsletter.mockResolvedValueOnce()
|
|
||||||
|
|
||||||
render(
|
|
||||||
<EditNewsletterDialog
|
|
||||||
newsletter={mockNewsletter}
|
|
||||||
isOpen={true}
|
|
||||||
folderOptions={["INBOX", "Archive"]}
|
|
||||||
onOpenChange={handleOpenChange}
|
|
||||||
onSuccess={handleSuccess}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Delete Newsletter/i }))
|
|
||||||
|
|
||||||
expect(window.confirm).toHaveBeenCalledWith('Are you sure you want to delete the "Existing Newsletter" newsletter?')
|
|
||||||
|
|
||||||
await waitFor(() => {
|
|
||||||
expect(mockedApi.deleteNewsletter).toHaveBeenCalledWith(1)
|
|
||||||
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
|
||||||
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
it("does not call deleteNewsletter when delete is not confirmed", () => {
|
|
||||||
window.confirm = jest.fn(() => false) // User clicks "Cancel"
|
|
||||||
|
|
||||||
render(
|
|
||||||
<EditNewsletterDialog
|
|
||||||
newsletter={mockNewsletter}
|
|
||||||
isOpen={true}
|
|
||||||
folderOptions={["INBOX", "Archive"]}
|
|
||||||
onOpenChange={handleOpenChange}
|
|
||||||
onSuccess={handleSuccess}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
|
|
||||||
fireEvent.click(screen.getByRole("button", { name: /Delete Newsletter/i }))
|
|
||||||
|
|
||||||
expect(mockedApi.deleteNewsletter).not.toHaveBeenCalled()
|
|
||||||
expect(handleSuccess).not.toHaveBeenCalled()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
@@ -0,0 +1,136 @@
|
|||||||
|
import React from "react"
|
||||||
|
import { render, screen, fireEvent, waitFor } from "@testing-library/react"
|
||||||
|
import "@testing-library/jest-dom"
|
||||||
|
import { NewsletterDialog } from "../NewsletterDialog"
|
||||||
|
import { Newsletter } from "@/lib/api"
|
||||||
|
import * as api from "@/lib/api"
|
||||||
|
|
||||||
|
// Mock the API module
|
||||||
|
jest.mock("@/lib/api", () => ({
|
||||||
|
...jest.requireActual("@/lib/api"),
|
||||||
|
createNewsletter: jest.fn(),
|
||||||
|
updateNewsletter: jest.fn(),
|
||||||
|
deleteNewsletter: jest.fn(),
|
||||||
|
}))
|
||||||
|
|
||||||
|
const mockedApi = api as jest.Mocked<typeof api>
|
||||||
|
|
||||||
|
const mockNewsletter: Newsletter = {
|
||||||
|
id: "1",
|
||||||
|
name: "Existing Newsletter",
|
||||||
|
is_active: true,
|
||||||
|
extract_content: false,
|
||||||
|
senders: [{ id: "1", email: "current@example.com" }],
|
||||||
|
entries_count: 5,
|
||||||
|
move_to_folder: "",
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("NewsletterDialog", () => {
|
||||||
|
const handleOpenChange = jest.fn()
|
||||||
|
const handleSuccess = jest.fn()
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks()
|
||||||
|
window.confirm = jest.fn(() => true)
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("Add Mode", () => {
|
||||||
|
it("allows user to fill out the form and submit", async () => {
|
||||||
|
mockedApi.createNewsletter.mockResolvedValueOnce({
|
||||||
|
id: "2",
|
||||||
|
name: "My New Newsletter",
|
||||||
|
is_active: true,
|
||||||
|
extract_content: false,
|
||||||
|
senders: [{ id: "2", email: "test@example.com" }],
|
||||||
|
entries_count: 0,
|
||||||
|
})
|
||||||
|
|
||||||
|
render(<NewsletterDialog isOpen={true} folderOptions={["INBOX", "Archive"]} onOpenChange={handleOpenChange} onSuccess={handleSuccess} />)
|
||||||
|
|
||||||
|
expect(screen.getByText("Register New Newsletter")).toBeInTheDocument()
|
||||||
|
|
||||||
|
fireEvent.change(screen.getByLabelText(/Newsletter Name/i), { target: { value: "My New Newsletter" } })
|
||||||
|
fireEvent.change(screen.getByPlaceholderText(/Enter email address/i), { target: { value: "test@example.com" } })
|
||||||
|
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Register Newsletter/i }))
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(mockedApi.createNewsletter).toHaveBeenCalledWith({
|
||||||
|
name: "My New Newsletter",
|
||||||
|
sender_emails: ["test@example.com"],
|
||||||
|
move_to_folder: "",
|
||||||
|
extract_content: false,
|
||||||
|
})
|
||||||
|
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
||||||
|
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("allows adding and removing email fields", () => {
|
||||||
|
render(<NewsletterDialog isOpen={true} folderOptions={[]} onOpenChange={() => {}} onSuccess={() => {}} />)
|
||||||
|
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(1)
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Add Another Email/i }))
|
||||||
|
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(2)
|
||||||
|
fireEvent.click(screen.getAllByRole("button", { name: /Remove/i })[0])
|
||||||
|
expect(screen.getAllByPlaceholderText(/Enter email address/i)).toHaveLength(1)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("Edit Mode", () => {
|
||||||
|
it("renders with initial newsletter data and allows updates", async () => {
|
||||||
|
mockedApi.updateNewsletter.mockResolvedValueOnce({ ...mockNewsletter, name: "Updated Name" })
|
||||||
|
|
||||||
|
render(
|
||||||
|
<NewsletterDialog
|
||||||
|
newsletter={mockNewsletter}
|
||||||
|
isOpen={true}
|
||||||
|
folderOptions={["INBOX", "Archive"]}
|
||||||
|
onOpenChange={handleOpenChange}
|
||||||
|
onSuccess={handleSuccess}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(screen.getByText("Edit Newsletter")).toBeInTheDocument()
|
||||||
|
const nameInput = screen.getByLabelText(/Newsletter Name/i)
|
||||||
|
expect(nameInput).toHaveValue("Existing Newsletter")
|
||||||
|
expect(screen.getByDisplayValue("current@example.com")).toBeInTheDocument()
|
||||||
|
|
||||||
|
fireEvent.change(nameInput, { target: { value: "Updated Name" } })
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Save Changes/i }))
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(mockedApi.updateNewsletter).toHaveBeenCalledWith("1", {
|
||||||
|
name: "Updated Name",
|
||||||
|
sender_emails: ["current@example.com"],
|
||||||
|
move_to_folder: "",
|
||||||
|
extract_content: false,
|
||||||
|
})
|
||||||
|
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
||||||
|
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("calls deleteNewsletter when delete button is clicked and confirmed", async () => {
|
||||||
|
mockedApi.deleteNewsletter.mockResolvedValueOnce(undefined)
|
||||||
|
|
||||||
|
render(
|
||||||
|
<NewsletterDialog
|
||||||
|
newsletter={mockNewsletter}
|
||||||
|
isOpen={true}
|
||||||
|
folderOptions={["INBOX", "Archive"]}
|
||||||
|
onOpenChange={handleOpenChange}
|
||||||
|
onSuccess={handleSuccess}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Delete Newsletter/i }))
|
||||||
|
expect(window.confirm).toHaveBeenCalledWith('Are you sure you want to delete the "Existing Newsletter" newsletter?')
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(mockedApi.deleteNewsletter).toHaveBeenCalledWith("1")
|
||||||
|
expect(handleSuccess).toHaveBeenCalledTimes(1)
|
||||||
|
expect(handleOpenChange).toHaveBeenCalledWith(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -12,8 +12,7 @@ import { LoadingSpinner } from "@/components/letterfeed/LoadingSpinner"
|
|||||||
import { Header } from "@/components/letterfeed/Header"
|
import { Header } from "@/components/letterfeed/Header"
|
||||||
import { NewsletterList } from "@/components/letterfeed/NewsletterList"
|
import { NewsletterList } from "@/components/letterfeed/NewsletterList"
|
||||||
import { EmptyState } from "@/components/letterfeed/EmptyState"
|
import { EmptyState } from "@/components/letterfeed/EmptyState"
|
||||||
import { AddNewsletterDialog } from "@/components/letterfeed/AddNewsletterDialog"
|
import { NewsletterDialog } from "@/components/letterfeed/NewsletterDialog"
|
||||||
import { EditNewsletterDialog } from "@/components/letterfeed/EditNewsletterDialog"
|
|
||||||
import { SettingsDialog } from "@/components/letterfeed/SettingsDialog"
|
import { SettingsDialog } from "@/components/letterfeed/SettingsDialog"
|
||||||
|
|
||||||
export default function LetterFeedApp() {
|
export default function LetterFeedApp() {
|
||||||
@@ -24,7 +23,6 @@ export default function LetterFeedApp() {
|
|||||||
|
|
||||||
const [isAddDialogOpen, setIsAddDialogOpen] = useState(false)
|
const [isAddDialogOpen, setIsAddDialogOpen] = useState(false)
|
||||||
const [isSettingsOpen, setIsSettingsOpen] = useState(false)
|
const [isSettingsOpen, setIsSettingsOpen] = useState(false)
|
||||||
const [isEditDialogOpen, setIsEditDialogOpen] = useState(false)
|
|
||||||
const [editingNewsletter, setEditingNewsletter] = useState<Newsletter | null>(null)
|
const [editingNewsletter, setEditingNewsletter] = useState<Newsletter | null>(null)
|
||||||
|
|
||||||
const fetchData = useCallback(async () => {
|
const fetchData = useCallback(async () => {
|
||||||
@@ -50,7 +48,10 @@ export default function LetterFeedApp() {
|
|||||||
|
|
||||||
const openEditDialog = (newsletter: Newsletter) => {
|
const openEditDialog = (newsletter: Newsletter) => {
|
||||||
setEditingNewsletter(newsletter)
|
setEditingNewsletter(newsletter)
|
||||||
setIsEditDialogOpen(true)
|
}
|
||||||
|
|
||||||
|
const closeEditDialog = () => {
|
||||||
|
setEditingNewsletter(null)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
@@ -71,25 +72,23 @@ export default function LetterFeedApp() {
|
|||||||
<EmptyState onAddNewsletter={() => setIsAddDialogOpen(true)} />
|
<EmptyState onAddNewsletter={() => setIsAddDialogOpen(true)} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<AddNewsletterDialog
|
<NewsletterDialog
|
||||||
isOpen={isAddDialogOpen}
|
isOpen={isAddDialogOpen}
|
||||||
folderOptions={folderOptions}
|
folderOptions={folderOptions}
|
||||||
onOpenChange={setIsAddDialogOpen}
|
onOpenChange={setIsAddDialogOpen}
|
||||||
onSuccess={fetchData}
|
onSuccess={fetchData}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{editingNewsletter && (
|
<NewsletterDialog
|
||||||
<EditNewsletterDialog
|
newsletter={editingNewsletter}
|
||||||
newsletter={editingNewsletter}
|
isOpen={!!editingNewsletter}
|
||||||
isOpen={isEditDialogOpen}
|
folderOptions={folderOptions}
|
||||||
folderOptions={folderOptions}
|
onOpenChange={closeEditDialog}
|
||||||
onOpenChange={setIsEditDialogOpen}
|
onSuccess={() => {
|
||||||
onSuccess={() => {
|
closeEditDialog()
|
||||||
setEditingNewsletter(null)
|
fetchData()
|
||||||
fetchData()
|
}}
|
||||||
}}
|
/>
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{settings && (
|
{settings && (
|
||||||
<SettingsDialog
|
<SettingsDialog
|
||||||
|
|||||||
Reference in New Issue
Block a user