"use client" import { useEffect, useRef, useState, useTransition } from "react" import { useSearchParams } from "next/navigation" import { PresetName, defaultPreset, getPreset } from "@/app/engine/presets" import { cn } from "@/lib/utils" import { TopMenu } from "./interface/top-menu" import { FontName, defaultFont, fonts } from "@/lib/fonts" import { getRandomLayoutName } from "./layouts" import { useStore } from "./store" import { Zoom } from "./interface/zoom" import { getStory } from "./queries/getStory" import { BottomBar } from "./interface/bottom-bar" import { Page } from "./interface/page" export default function Main() { const [_isPending, startTransition] = useTransition() const searchParams = useSearchParams() const requestedPreset = (searchParams.get('preset') as PresetName) || defaultPreset const requestedFont = (searchParams.get('font') as FontName) || defaultFont const requestedPrompt = (searchParams.get('prompt') as string) || "" const isGeneratingStory = useStore(state => state.isGeneratingStory) const setGeneratingStory = useStore(state => state.setGeneratingStory) const font = useStore(state => state.font) const setFont = useStore(state => state.setFont) const preset = useStore(state => state.preset) const setPreset = useStore(state => state.setPreset) const prompt = useStore(state => state.prompt) const setPrompt = useStore(state => state.setPrompt) const setLayouts = useStore(state => state.setLayouts) const setPanels = useStore(state => state.setPanels) const zoomLevel = useStore(state => state.zoomLevel) const [waitABitMore, setWaitABitMore] = useState(false) // react to URL params useEffect(() => { if (requestedPreset && requestedPreset !== preset.label) { setPreset(getPreset(requestedPreset)) } }, [requestedPreset]) useEffect(() => { if (requestedFont && requestedFont !== font) { setFont(requestedFont) } }, [requestedFont]) useEffect(() => { if (requestedPrompt && requestedPrompt !== prompt) { setPrompt(requestedPrompt) } }, [requestedPrompt]) // react to prompt changes useEffect(() => { if (!prompt) { return } startTransition(async () => { setWaitABitMore(false) setGeneratingStory(true) const newLayouts = [ getRandomLayoutName(), getRandomLayoutName(), ] console.log("using layouts " + newLayouts) setLayouts(newLayouts) try { const llmResponse = await getStory({ preset, prompt }) console.log("response:", llmResponse) const panelPromptPrefix = preset.imagePrompt(prompt).join(", ") console.log("panel prompt prefix:", panelPromptPrefix) const nbPanels = 4 const newPanels: string[] = [] setWaitABitMore(true) for (let p = 0; p < nbPanels; p++) { const newPanel = [panelPromptPrefix, llmResponse[p] || ""] newPanels.push(newPanel.map(chunk => chunk).join(", ")) } console.log("newPanels:", newPanels) setPanels(newPanels) } catch (err) { console.error(err) } finally { setTimeout(() => { setGeneratingStory(false) setWaitABitMore(false) }, 12000) } }) }, [prompt, preset?.label]) // important: we need to react to preset changes too return (
105 ? `px-0` : `pl-2 pr-16 md:pl-16 md:pr-16`, `print:pt-0 print:px-0 print:pl-0 print:pr-0`, fonts.actionman.className )}>
105 ? `items-start` : `items-center` )}>
{/* // we could support multiple pages here, // but let's disable it for now */}
{waitABitMore ? `Story is ready, but server is a bit busy!`: 'Generating a new story..'}
{waitABitMore ? `Please hold tight..` : ''}
) }