import { Show, createSignal, createEffect, onCleanup } from "solid-js"; import { useNavigate } from "@solidjs/router"; import { api } from "~/lib/api"; import { debounce } from "es-toolkit"; import Dropzone from "~/components/blog/Dropzone"; import TextEditor from "~/components/blog/TextEditor"; import TagMaker from "~/components/blog/TagMaker"; import AddAttachmentSection from "~/components/blog/AddAttachmentSection"; import XCircle from "~/components/icons/XCircle"; import AddImageToS3 from "~/lib/s3upload"; interface PostFormProps { mode: "create" | "edit"; postId?: number; initialData?: { title: string; subtitle: string; body: string; banner_photo: string; published: boolean; tags: string[]; }; userID: number; } export default function PostForm(props: PostFormProps) { const navigate = useNavigate(); const [title, setTitle] = createSignal(props.initialData?.title || ""); const [subtitle, setSubtitle] = createSignal( props.initialData?.subtitle || "" ); const [body, setBody] = createSignal(props.initialData?.body || ""); const [bannerPhoto, setBannerPhoto] = createSignal( props.initialData?.banner_photo || "" ); const [bannerImageFile, setBannerImageFile] = createSignal(); const [bannerImageHolder, setBannerImageHolder] = createSignal< string | ArrayBuffer | null >(null); const [requestedDeleteImage, setRequestedDeleteImage] = createSignal(false); const [published, setPublished] = createSignal( props.initialData?.published || false ); const [tags, setTags] = createSignal(props.initialData?.tags || []); const [tagInputValue, setTagInputValue] = createSignal(""); const [loading, setLoading] = createSignal(false); const [error, setError] = createSignal(""); const [showAutoSaveMessage, setShowAutoSaveMessage] = createSignal(false); const [isInitialLoad, setIsInitialLoad] = createSignal(props.mode === "edit"); const [initialBody, setInitialBody] = createSignal( props.initialData?.body ); const [hasSaved, setHasSaved] = createSignal(props.mode === "edit"); // Mark initial load as complete after data is loaded (for edit mode) createEffect(() => { if (props.mode === "edit" && props.initialData) { setIsInitialLoad(false); } }); const autoSave = async () => { const titleVal = title(); if (titleVal) { try { let bannerImageKey = ""; const bannerFile = bannerImageFile(); if (bannerFile) { bannerImageKey = (await AddImageToS3( bannerFile, titleVal, "blog" )) as string; } if (props.mode === "edit") { await api.database.updatePost.mutate({ id: props.postId!, title: titleVal.replaceAll(" ", "_"), subtitle: subtitle() || null, body: body() || null, banner_photo: bannerImageKey !== "" ? bannerImageKey : requestedDeleteImage() ? "_DELETE_IMAGE_" : null, published: published(), tags: tags().length > 0 ? tags() : null, author_id: props.userID }); } else { // Create mode: only save once if (!hasSaved()) { await api.database.createPost.mutate({ category: "blog", title: titleVal.replaceAll(" ", "_"), subtitle: subtitle() || null, body: body() || null, banner_photo: bannerImageKey !== "" ? bannerImageKey : null, published: published(), tags: tags().length > 0 ? tags() : null, author_id: props.userID }); setHasSaved(true); } } showAutoSaveTrigger(); } catch (err) { console.error("Autosave failed:", err); } } }; const showAutoSaveTrigger = () => { setShowAutoSaveMessage(true); setTimeout(() => { setShowAutoSaveMessage(false); }, 5000); }; // Debounced auto-save (1 second after last change) const debouncedAutoSave = debounce(autoSave, 1000); // Track changes to trigger auto-save createEffect(() => { const titleVal = title(); const subtitleVal = subtitle(); const bodyVal = body(); const tagsVal = tags(); const publishedVal = published(); // Only trigger auto-save if conditions are met if (props.mode === "edit" && !isInitialLoad() && titleVal) { debouncedAutoSave(); } else if (props.mode === "create" && titleVal) { debouncedAutoSave(); } }); onCleanup(() => { debouncedAutoSave.cancel(); }); const handleBannerImageDrop = (acceptedFiles: File[]) => { if (acceptedFiles.length > 0) { const file = acceptedFiles[0]; if (props.mode === "edit") { setRequestedDeleteImage(false); } setBannerImageFile(file); const reader = new FileReader(); reader.onload = () => { setBannerImageHolder(reader.result); }; reader.readAsDataURL(file); } }; const removeBannerImage = () => { setBannerImageFile(undefined); setBannerImageHolder(null); if (props.mode === "edit") { setRequestedDeleteImage(true); } }; const tagHandler = (input: string) => { const split = input.split(" "); if (split.length > 1) { const newSplit: string[] = []; split.forEach((word) => { if (word[0] === "#" && word.length > 1) { setTags((prevTags) => [...prevTags, word]); } else { newSplit.push(word); } }); setTagInputValue(newSplit.join(" ")); } else { setTagInputValue(input); } }; const deleteTag = (idx: number) => { setTags((prevTags) => prevTags.filter((_, index) => index !== idx)); }; const handleSubmit = async (e: Event) => { e.preventDefault(); if (!props.userID) { setError(`You must be logged in to ${props.mode} posts`); return; } if (!title()) { setError("Title is required to publish"); return; } setLoading(true); setError(""); try { let bannerImageKey = ""; const bannerFile = bannerImageFile(); if (bannerFile) { bannerImageKey = (await AddImageToS3( bannerFile, title(), "blog" )) as string; } if (props.mode === "edit") { await api.database.updatePost.mutate({ id: props.postId!, title: title().replaceAll(" ", "_"), subtitle: subtitle() || null, body: body() || null, banner_photo: bannerImageKey !== "" ? bannerImageKey : requestedDeleteImage() ? "_DELETE_IMAGE_" : null, published: published(), tags: tags().length > 0 ? tags() : null, author_id: props.userID }); } else { await api.database.createPost.mutate({ category: "blog", title: title().replaceAll(" ", "_"), subtitle: subtitle() || null, body: body() || null, banner_photo: bannerImageKey !== "" ? bannerImageKey : null, published: published(), tags: tags().length > 0 ? tags() : null, author_id: props.userID }); } navigate(`/blog/${encodeURIComponent(title().replaceAll(" ", "_"))}`); } catch (err) { console.error(`Error ${props.mode}ing post:`, err); setError(`Failed to ${props.mode} post. Please try again.`); } finally { setLoading(false); } }; return (
{props.mode === "edit" ? "Edit a Blog" : "Create a Blog"}
{/* Title */}
setTitle(e.currentTarget.value)} name="title" placeholder=" " class="underlinedInput w-full bg-transparent" />
{/* Subtitle */}
setSubtitle(e.currentTarget.value)} name="subtitle" placeholder=" " class="underlinedInput w-full bg-transparent" />
{/* Banner */}
Banner
{/* Attachments */} {/* Text Editor */}
{/* Tags */} {/* Auto-save message */}
{showAutoSaveMessage() ? "Auto save success!" : ""}
{/* Publish checkbox */}
setPublished(e.currentTarget.checked)} />
Published
{/* Error message */}
{error()}
{/* Submit button */}
); }