File size: 5,262 Bytes
624088c dbc8f44 932a7fd 624088c 932a7fd 624088c f4dea7d 932a7fd 8c5d17c f4dea7d dbc8f44 624088c 932a7fd 624088c f4dea7d 932a7fd 624088c 932a7fd 624088c 932a7fd 624088c 8c5d17c dbc8f44 932a7fd f4dea7d a79c634 624088c 932a7fd 624088c 8c5d17c 932a7fd 8c5d17c f4dea7d a40bf40 f4dea7d 8c5d17c dbc8f44 8c5d17c a40bf40 82d85df 8c5d17c a40bf40 8c5d17c 82d85df 8c5d17c 932a7fd 8c5d17c dbc8f44 f4dea7d a40bf40 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 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 |
"use client"
import { useEffect, useRef, 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, fontList, fonts } from "@/lib/fonts"
import { getRandomLayoutName, layouts } from "./layouts"
import { useStore } from "./store"
import { Zoom } from "./interface/zoom"
import { getStory } from "./queries/getStory"
import { BottomBar } from "./interface/bottom-bar"
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 layout = useStore(state => state.layout)
const setLayout = useStore(state => state.setLayout)
const setPanels = useStore(state => state.setPanels)
const zoomLevel = useStore(state => state.zoomLevel)
const setPage = useStore(state => state.setPage)
const pageRef = useRef<HTMLDivElement>(null)
useEffect(() => {
const element = pageRef.current
if (!element) { return }
setPage(element)
}, [pageRef.current])
// 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 () => {
setGeneratingStory(true)
const newLayout = getRandomLayoutName()
console.log("using layout " + newLayout)
setLayout(newLayout)
try {
const llmResponse = await getStory({ preset, prompt })
console.log("response:", llmResponse)
// TODO call the LLM here!
const panelPromptPrefix = preset.imagePrompt(prompt).join(", ")
console.log("panel prompt prefix:", panelPromptPrefix)
const nbPanels = 4
const newPanels: string[] = []
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 {
setGeneratingStory(false)
}
})
}, [prompt, preset?.label]) // important: we need to react to preset changes too
const LayoutElement = (layouts as any)[layout]
return (
<div>
<TopMenu />
<div className={cn(
`flex items-start w-screen h-screen pt-[120px] px-16 md:pt-[72px] overflow-y-scroll`,
`transition-all duration-200 ease-in-out`,
`print:pt-2 print:px-2`,
fonts.actionman.className
)}>
<div className="flex flex-col items-center w-full">
<div
ref={pageRef}
className={cn(
`comic-page`,
`flex flex-col items-center justify-start`,
// we are trying to reach a "book" look
// we are using aspect-[297/210] because it matches A4 (297mm x 210mm)
// `aspect-[210/297]`,
`aspect-[250/297]`,
`transition-all duration-100 ease-in-out`,
`border border-stone-200`,
`shadow-2xl`,
`print:shadow-none`,
`print:width-screen`
)}
style={{
width: `${zoomLevel}%`,
padding: `${Math.round((zoomLevel / 100) * 16)}px`
}}
>
<LayoutElement />
</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-lg text-stone-600 w-[70%]`,
isGeneratingStory ? ``: `scale-0 opacity-0`,
`transition-all duration-300 ease-in-out`,
)}>
Generating your story.. (hold tight)
</div>
</div>
</div>
)
} |