import { Show, createSignal, createEffect, onCleanup } from "solid-js"; import { useNavigate, query } from "@solidjs/router"; import { Title } from "@solidjs/meta"; import { createAsync } from "@solidjs/router"; import { getRequestEvent } from "solid-js/web"; import { api } from "~/lib/api"; import { debounce } from "~/lib/client-utils"; 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"; const getAuthState = query(async () => { "use server"; const { getPrivilegeLevel, getUserID } = await import("~/server/utils"); const event = getRequestEvent()!; const privilegeLevel = await getPrivilegeLevel(event.nativeEvent); const userID = await getUserID(event.nativeEvent); return { privilegeLevel, userID }; }, "auth-state"); export default function CreatePost() { const navigate = useNavigate(); const authState = createAsync(() => getAuthState()); const [title, setTitle] = createSignal(""); const [subtitle, setSubtitle] = createSignal(""); const [body, setBody] = createSignal(""); const [bannerPhoto, setBannerPhoto] = createSignal(""); const [bannerImageFile, setBannerImageFile] = createSignal(); const [bannerImageHolder, setBannerImageHolder] = createSignal< string | ArrayBuffer | null >(null); const [published, setPublished] = createSignal(false); const [tags, setTags] = createSignal([]); const [tagInputValue, setTagInputValue] = createSignal(""); const [loading, setLoading] = createSignal(false); const [error, setError] = createSignal(""); const [showAutoSaveMessage, setShowAutoSaveMessage] = createSignal(false); const [hasSaved, setHasSaved] = createSignal(false); const autoSave = async () => { const titleVal = title(); const bodyVal = body(); if (titleVal && bodyVal !== "") { try { let bannerImageKey = ""; const bannerFile = bannerImageFile(); if (bannerFile) { bannerImageKey = (await AddImageToS3( bannerFile, titleVal, "blog" )) as string; } const method = hasSaved() ? "PATCH" : "POST"; if (method === "POST") { await api.database.createPost.mutate({ category: "blog", title: titleVal.replaceAll(" ", "_"), subtitle: subtitle() || null, body: bodyVal || null, banner_photo: bannerImageKey !== "" ? bannerImageKey : null, published: published(), tags: tags().length > 0 ? tags() : null, author_id: authState()!.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(() => { // Track all relevant fields const titleVal = title(); const subtitleVal = subtitle(); const bodyVal = body(); const tagsVal = tags(); const publishedVal = published(); // Only trigger auto-save if we have at least title and body if (titleVal && bodyVal) { debouncedAutoSave(); } }); onCleanup(() => { debouncedAutoSave.cancel(); }); const handleBannerImageDrop = (acceptedFiles: File[]) => { if (acceptedFiles.length > 0) { const file = acceptedFiles[0]; setBannerImageFile(file); const reader = new FileReader(); reader.onload = () => { setBannerImageHolder(reader.result); }; reader.readAsDataURL(file); } }; const removeBannerImage = () => { setBannerImageFile(undefined); setBannerImageHolder(null); }; 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 (!authState()?.userID) { setError("You must be logged in to create a post"); return; } setLoading(true); setError(""); try { let bannerImageKey = ""; const bannerFile = bannerImageFile(); if (bannerFile) { bannerImageKey = (await AddImageToS3( bannerFile, title(), "blog" )) as string; } const result = 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: authState()!.userID }); if (result.data) { navigate(`/blog/${encodeURIComponent(title().replaceAll(" ", "_"))}`); } } catch (err) { console.error("Error creating post:", err); setError("Failed to create post. Please try again."); } finally { setLoading(false); } }; return ( <> Create Blog Post | Michael Freno
Unauthorized
You must be an admin to create posts.
} >
Create a Blog
{/* Title */}
setTitle(e.currentTarget.value)} required 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)} />
Publish
{/* Error message */}
{error()}
{/* Submit button */}
); }