File size: 4,346 Bytes
624088c 797cbb8 624088c 797cbb8 932a7fd 8c5d17c f4dea7d dbc8f44 cb3fdda 624088c f4dea7d 932a7fd cb3fdda 624088c 932a7fd c32ec0d 624088c 8c5d17c cb3fdda 932a7fd f4dea7d cb3fdda f4dea7d cb3fdda f4dea7d c380867 f4dea7d c32ec0d cb3fdda f4dea7d c32ec0d f4dea7d c32ec0d f4dea7d cb3fdda 66d8bb7 f4dea7d a79c634 624088c 8c5d17c 932a7fd 8c5d17c 974ed41 f4dea7d 974ed41 f4dea7d 8c5d17c cb3fdda 8c5d17c a40bf40 cb3fdda 8c5d17c 82d85df cb3fdda 8c5d17c 932a7fd 8c5d17c dbc8f44 f4dea7d a40bf40 f4dea7d cb3fdda f4dea7d cb3fdda f4dea7d 624088c |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 |
"use client"
import { useEffect, useState, useTransition } from "react"
import { cn } from "@/lib/utils"
import { TopMenu } from "./interface/top-menu"
import { fonts } from "@/lib/fonts"
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 isGeneratingStory = useStore(state => state.isGeneratingStory)
const setGeneratingStory = useStore(state => state.setGeneratingStory)
const font = useStore(state => state.font)
const preset = useStore(state => state.preset)
const prompt = useStore(state => state.prompt)
const setLayouts = useStore(state => state.setLayouts)
const setPanels = useStore(state => state.setPanels)
const setCaptions = useStore(state => state.setCaptions)
const zoomLevel = useStore(state => state.zoomLevel)
const [waitABitMore, setWaitABitMore] = useState(false)
// react to prompt changes
useEffect(() => {
if (!prompt) { return }
startTransition(async () => {
setWaitABitMore(false)
setGeneratingStory(true)
try {
const llmResponse = await getStory({ preset, prompt })
console.log("response:", llmResponse)
// we have to limit the size of the prompt, otherwise the rest of the style won't be followed
let limitedPrompt = prompt.slice(0, 77)
console.log("had to cut the length og the prompt to:", limitedPrompt)
const panelPromptPrefix = preset.imagePrompt(limitedPrompt).join(", ")
console.log("panel prompt prefix:", panelPromptPrefix)
const nbPanels = 4
const newPanels: string[] = []
const newCaptions: string[] = []
setWaitABitMore(true)
for (let p = 0; p < nbPanels; p++) {
newCaptions.push(llmResponse[p]?.caption || "...")
const newPanel = [panelPromptPrefix, llmResponse[p]?.instructions || ""]
newPanels.push(newPanel.map(chunk => chunk).join(", "))
}
console.log("newPanels:", newPanels)
setCaptions(newCaptions)
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 (
<div>
<TopMenu />
<div className={cn(
`flex items-start w-screen h-screen pt-[120px] md:pt-[72px] overflow-y-scroll`,
`transition-all duration-200 ease-in-out`,
zoomLevel > 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
)}>
<div
className={cn(
`flex flex-col w-full`,
zoomLevel > 105 ? `items-start` : `items-center`
)}>
<div
className={cn(
`comic-page`,
`flex flex-col md:flex-row md:space-x-16 md:items-center md:justify-start`,
)}
style={{
width: `${zoomLevel}%`
}}>
<Page page={0} />
{/*
// we could support multiple pages here,
// but let's disable it for now
<Page page={1} />
*/}
</div>
</div>
</div>
<Zoom />
<BottomBar />
<div className={cn(
`print:hidden`,
`z-20 fixed inset-0`,
`flex flex-row items-center justify-center`,
`transition-all duration-300 ease-in-out`,
isGeneratingStory
? `bg-zinc-100/10 backdrop-blur-md`
: `bg-zinc-100/0 backdrop-blur-none pointer-events-none`,
fonts.actionman.className
)}>
<div className={cn(
`text-center text-xl text-stone-600 w-[70%]`,
isGeneratingStory ? ``: `scale-0 opacity-0`,
`transition-all duration-300 ease-in-out`,
)}>
{waitABitMore ? `Story is ready, but server is a bit busy!`: 'Generating a new story..'}<br/>
{waitABitMore ? `Please hold tight..` : ''}
</div>
</div>
</div>
)
} |