jbilcke-hf's picture
jbilcke-hf HF staff
show speech bubbles by default
2e12a66
"use client"
import { create } from "zustand"
import { ClapProject, ClapMediaOrientation, ClapSegment, ClapSegmentCategory, ClapSegmentStatus, ClapOutputType, ClapSegmentFilteringMode, filterSegments, newClap, newSegment, parseClap, serializeClap } from "@aitube/clap"
import { FontName } from "@/lib/fonts"
import { Preset, PresetName, defaultPreset, getPreset, getRandomPreset } from "@/app/engine/presets"
import { RenderedScene } from "@/types"
import { getParam } from "@/lib/getParam"
import { LayoutName, defaultLayout, getRandomLayoutName } from "../layouts"
import { putTextInInput } from "@/lib/putTextInInput"
import { parsePresetFromPrompts } from "@/lib/parsePresetFromPrompts"
import { parseLayoutFromStoryboards } from "@/lib/parseLayoutFromStoryboards"
import { getLocalStorageShowSpeeches } from "@/lib/getLocalStorageShowSpeeches"
export const useStore = create<{
prompt: string
font: FontName
preset: Preset
currentClap?: ClapProject
currentNbPanelsPerPage: number
maxNbPanelsPerPage: number
currentNbPages: number
maxNbPages: number
previousNbPanels: number
currentNbPanels: number
maxNbPanels: number
panels: string[]
speeches: string[]
captions: string[]
upscaleQueue: Record<string, RenderedScene>
showSpeeches: boolean
showCaptions: boolean
renderedScenes: Record<string, RenderedScene>
layout: LayoutName
layouts: LayoutName[]
zoomLevel: number
page: HTMLDivElement
isGeneratingStory: boolean
panelGenerationStatus: Record<number, boolean>
isGeneratingText: boolean
atLeastOnePanelIsBusy: boolean
setCurrentNbPanelsPerPage: (currentNbPanelsPerPage: number) => void
setMaxNbPanelsPerPage: (maxNbPanelsPerPage: number) => void
setCurrentNbPages: (currentNbPages: number) => void
setMaxNbPages: (maxNbPages: number) => void
setPreviousNbPanels: (previousNbPanels: number) => void
setCurrentNbPanels: (currentNbPanels: number) => void
setMaxNbPanels: (maxNbPanels: number) => void
setRendered: (panelId: string, renderedScene: RenderedScene) => void
addToUpscaleQueue: (panelId: string, renderedScene: RenderedScene) => void
removeFromUpscaleQueue: (panelId: string) => void
setPrompt: (prompt: string) => void
setFont: (font: FontName) => void
setPreset: (preset: Preset) => void
setPanels: (panels: string[]) => void
setPanelPrompt: (newPrompt: string, index: number) => void
setLayout: (layout: LayoutName, index?: number) => void
setLayouts: (layouts: LayoutName[]) => void
setShowSpeeches: (showSpeeches: boolean) => void
setSpeeches: (speeches: string[]) => void
setPanelSpeech: (newSpeech: string, index: number) => void
setShowCaptions: (showCaptions: boolean) => void
setCaptions: (captions: string[]) => void
setPanelCaption: (newCaption: string, index: number) => void
setZoomLevel: (zoomLevel: number) => void
setGeneratingStory: (isGeneratingStory: boolean) => void
setGeneratingImages: (panelId: string, value: boolean) => void
setGeneratingText: (isGeneratingText: boolean) => void
// I think we should deprecate those three functions
// this was used to keep track of the page HTML element,
// for use with a HTML-to-bitmap library
// but the CSS layout wasn't followed properly and it depended on the zoom level
// pageToImage: () => Promise<string>
// download: () => Promise<void>
// setPage: (page: HTMLDivElement) => void
generate: (prompt: string, presetName: PresetName, layoutName: LayoutName) => void
convertComicToClap: () => Promise<ClapProject>
convertClapToComic: (clap: ClapProject) => Promise<{
currentNbPanels: number
prompt: string
preset: Preset
layout: LayoutName
storyPrompt: string
stylePrompt: string
panels: string[]
renderedScenes: Record<string, RenderedScene>
speeches: string[]
captions: string[]
}>
loadClap: (blob: Blob) => Promise<void>
downloadClap: () => Promise<void>
}>((set, get) => ({
// -------- note --------------------------------------------------
// do not read the local storage in this block, results might be empty
// ----------------------------------------------------------------
prompt:
(getParam("stylePrompt", "") || getParam("storyPrompt", ""))
? `${getParam("stylePrompt", "")}||${getParam("storyPrompt", "")}`
: "",
font: "actionman",
preset: getPreset(getParam("preset", defaultPreset)),
currentClap: undefined,
currentNbPanelsPerPage: 4,
maxNbPanelsPerPage: 4,
currentNbPages: 1,
maxNbPages: getParam("maxNbPages", 1),
previousNbPanels: 0,
currentNbPanels: 4,
maxNbPanels: 4,
panels: [],
speeches: [],
captions: [],
upscaleQueue: {} as Record<string, RenderedScene>,
renderedScenes: {} as Record<string, RenderedScene>,
showSpeeches: true,
showCaptions: getParam("showCaptions", false),
// deprecated?
layout: defaultLayout,
layouts: [defaultLayout, defaultLayout, defaultLayout, defaultLayout],
zoomLevel: getParam("zoomLevel", 60),
// deprecated?
page: undefined as unknown as HTMLDivElement,
isGeneratingStory: false,
panelGenerationStatus: {},
isGeneratingText: false,
atLeastOnePanelIsBusy: false,
setCurrentNbPanelsPerPage: (currentNbPanelsPerPage: number) => {
const { currentNbPages } = get()
set({
currentNbPanelsPerPage,
currentNbPanels: currentNbPanelsPerPage * currentNbPages
})
},
setMaxNbPanelsPerPage: (maxNbPanelsPerPage: number) => {
const { maxNbPages } = get()
set({
maxNbPanelsPerPage,
maxNbPanels: maxNbPanelsPerPage * maxNbPages,
})
},
setCurrentNbPages: (currentNbPages: number) => {
const state = get()
const newCurrentNumberOfPages = Math.min(state.maxNbPages, currentNbPages)
const newCurrentNbPanels = state.currentNbPanelsPerPage * newCurrentNumberOfPages
/*
console.log(`setCurrentNbPages(${currentNbPages}): ${JSON.stringify({
"state.maxNbPages": state.maxNbPages,
currentNbPages,
newCurrentNumberOfPages,
"state.currentNbPanelsPerPage": state.currentNbPanelsPerPage,
newCurrentNbPanels,
"state.currentNbPanels": state.currentNbPanels,
"state.previousNbPanels": state.previousNbPanels,
previousNbPanels:
newCurrentNbPanels > state.currentNbPanels ? state.currentNbPanels :
newCurrentNbPanels < state.currentNbPanels ? 0 :
state.previousNbPanels,
}, null, 2)}`)
*/
set({
// we keep the previous number of panels for convenience
// so if we are adding a new panel,
// state.currentNbPanels gets copied to state.previousNbPanels
previousNbPanels:
newCurrentNbPanels > state.currentNbPanels ? state.currentNbPanels :
newCurrentNbPanels < state.currentNbPanels ? 0 :
state.previousNbPanels,
currentNbPanels: newCurrentNbPanels,
currentNbPages: newCurrentNumberOfPages,
})
},
setMaxNbPages: (maxNbPages: number) => {
const { maxNbPanelsPerPage } = get()
set({
maxNbPages,
maxNbPanels: maxNbPanelsPerPage * maxNbPages,
})
},
setPreviousNbPanels: (previousNbPanels: number) => {
set({
previousNbPanels
})
},
setCurrentNbPanels: (currentNbPanels: number) => {
const state = get()
/*
console.log(`setCurrentNbPanels(${currentNbPanels}): ${JSON.stringify({
"state.maxNbPages": state.maxNbPages,
"state.currentNbPages": state.currentNbPages,
currentNbPanels,
"state.currentNbPanelsPerPage": state.currentNbPanelsPerPage,
"state.currentNbPanels": state.currentNbPanels,
"state.previousNbPanels": state.previousNbPanels,
previousNbPanels:
currentNbPanels > state.currentNbPanels ? state.currentNbPanels :
currentNbPanels < state.currentNbPanels ? 0 :
state.previousNbPanels,
}, null, 2)}`)
*/
set({
// we keep the previous number of panels for convenience
// so if we are adding a new panel,
// state.currentNbPanels gets copied to state.previousNbPanels
previousNbPanels:
currentNbPanels > state.currentNbPanels ? state.currentNbPanels :
currentNbPanels < state.currentNbPanels ? 0 :
state.previousNbPanels,
currentNbPanels,
})
},
setMaxNbPanels: (maxNbPanels: number) => {
set({
maxNbPanels
})
},
setRendered: (panelId: string, renderedScene: RenderedScene) => {
const { renderedScenes } = get()
set({
renderedScenes: {
...renderedScenes,
[panelId]: renderedScene
}
})
},
addToUpscaleQueue: (panelId: string, renderedScene: RenderedScene) => {
const { upscaleQueue } = get()
set({
upscaleQueue: {
...upscaleQueue,
[panelId]: renderedScene
},
})
},
removeFromUpscaleQueue: (panelId: string) => {
const upscaleQueue = { ...get().upscaleQueue }
delete upscaleQueue[panelId]
set({
upscaleQueue,
})
},
setPrompt: (prompt: string) => {
const existingPrompt = get().prompt
if (prompt === existingPrompt) { return }
set({
prompt,
})
},
setFont: (font: FontName) => {
const existingFont = get().font
if (font === existingFont) { return }
set({
font,
})
},
setPreset: (preset: Preset) => {
const existingPreset = get().preset
if (preset.label === existingPreset.label) { return }
set({
preset,
})
},
setPanels: (panels: string[]) => set({ panels }),
setPanelPrompt: (newPrompt, index) => {
const { panels } = get()
set({
panels: panels.map((p, i) => (
index === i ? newPrompt : p
))
})
},
setSpeeches: (speeches: string[]) => {
set({
speeches,
})
},
setShowSpeeches: (showSpeeches: boolean) => {
set({
showSpeeches,
})
try {
localStorage.setItem("AI_COMIC_FACTORY_SHOW_SPEECHES", `${showSpeeches || false}`)
} catch (err) {
console.error(`failed to persist "showSpeeches" for value "${showSpeeches}"`)
}
},
setPanelSpeech: (newSpeech, index) => {
const { speeches } = get()
set({
speeches: speeches.map((c, i) => (
index === i ? newSpeech : c
))
})
},
setCaptions: (captions: string[]) => {
set({
captions,
})
},
setShowCaptions: (showCaptions: boolean) => {
set({
showCaptions,
})
},
setPanelCaption: (newCaption, index) => {
const { captions } = get()
set({
captions: captions.map((c, i) => (
index === i ? newCaption : c
))
})
},
setLayout: (layoutName: LayoutName, index?: number) => {
const { maxNbPages, currentNbPanelsPerPage, layouts } = get()
for (let i = 0; i < maxNbPages; i++) {
let name = layoutName === "random" ? getRandomLayoutName() : layoutName
if (typeof index === "number" && !isNaN(index) && isFinite(index)) {
if (i === index) {
layouts[i] = name
}
} else {
layouts[i] = name
}
}
set({
// changing the layout isn't a free pass to generate tons of panels at once,
// so we reset pretty much everything
previousNbPanels: 0,
currentNbPages: 1,
currentNbPanels: currentNbPanelsPerPage,
panels: [],
speeches: [],
captions: [],
upscaleQueue: {},
renderedScenes: {},
isGeneratingStory: false,
panelGenerationStatus: {},
isGeneratingText: false,
atLeastOnePanelIsBusy: false,
layouts,
layout: layouts[0],
})
},
setLayouts: (layouts: LayoutName[]) => set({ layouts }),
setZoomLevel: (zoomLevel: number) => set({ zoomLevel }),
setGeneratingStory: (isGeneratingStory: boolean) => set({ isGeneratingStory }),
setGeneratingImages: (panelId: string, value: boolean) => {
const panelGenerationStatus: Record<string, boolean> = {
...get().panelGenerationStatus,
[panelId]: value
}
const atLeastOnePanelIsBusy = Object.values(panelGenerationStatus).includes(true)
set({
panelGenerationStatus,
atLeastOnePanelIsBusy
})
},
setGeneratingText: (isGeneratingText: boolean) => set({ isGeneratingText }),
// I think we should deprecate those three functions
// this was used to keep track of the page HTML element,
// for use with a HTML-to-bitmap library
// but the CSS layout wasn't followed properly and it depended on the zoom level
/*
setPage: (page: HTMLDivElement) => {
if (!page) { return }
set({ page })
},
pageToImage: async () => {
const { page } = get()
if (!page) { return "" }
const canvas = await html2canvas(page)
// console.log("canvas:", canvas)
const data = canvas.toDataURL('image/jpeg', 0.97)
return data
},
download: async () => {
const { pageToImage } = get()
const data = await pageToImage()
const link = document.createElement('a')
if (typeof link.download === 'string') {
link.href = data
link.download = 'comic.jpg'
document.body.appendChild(link)
link.click()
document.body.removeChild(link)
} else {
window.open(data)
}
},
*/
generate: (prompt: string, presetName: PresetName, layoutName: LayoutName) => {
const { maxNbPages, currentNbPanelsPerPage } = get()
const layouts: LayoutName[] = []
for (let i = 0; i < maxNbPages; i++) {
layouts.push(
layoutName === "random"
? getRandomLayoutName()
: layoutName)
}
set({
// we reset pretty much everything
previousNbPanels: 0,
currentNbPages: 1,
currentNbPanels: currentNbPanelsPerPage,
panels: [],
speeches: [],
captions: [],
upscaleQueue: {},
renderedScenes: {},
isGeneratingStory: false,
panelGenerationStatus: {},
isGeneratingText: false,
atLeastOnePanelIsBusy: false,
prompt,
preset: presetName === "random"
? getRandomPreset()
: getPreset(presetName),
layout: layouts[0],
layouts,
})
},
convertComicToClap: async (): Promise<ClapProject> => {
const {
currentNbPanels,
prompt,
panels,
renderedScenes,
speeches,
captions
} = get()
const defaultSegmentDurationInMs = 7000
let currentElapsedTimeInMs = 0
const clap: ClapProject = newClap({
meta: {
title: "Untitled", // we don't need a title actually
description: prompt,
prompt: prompt,
synopsis: "",
licence: "",
orientation: ClapMediaOrientation.LANDSCAPE,
width: 512,
height: 288,
isInteractive: false,
isLoop: false,
durationInMs: panels.length * defaultSegmentDurationInMs,
defaultVideoModel: "SDXL",
}
})
for (let i = 0; i < panels.length; i++) {
const panel = panels[i]
const speech = speeches[i]
const caption = captions[i]
const renderedScene = renderedScenes[`${i}`]
clap.segments.push(newSegment({
track: 1,
startTimeInMs: currentElapsedTimeInMs,
assetDurationInMs: defaultSegmentDurationInMs,
category: ClapSegmentCategory.STORYBOARD,
prompt: panel,
outputType: ClapOutputType.IMAGE,
assetUrl: renderedScene?.assetUrl || "",
status: ClapSegmentStatus.COMPLETED,
}))
clap.segments.push(newSegment({
track: 2,
startTimeInMs: currentElapsedTimeInMs,
assetDurationInMs: defaultSegmentDurationInMs,
category: ClapSegmentCategory.INTERFACE,
prompt: caption,
// assetUrl: `data:text/plain;base64,${btoa(title)}`,
assetUrl: caption,
outputType: ClapOutputType.TEXT,
status: ClapSegmentStatus.COMPLETED,
}))
clap.segments.push(newSegment({
track: 3,
startTimeInMs: currentElapsedTimeInMs,
assetDurationInMs: defaultSegmentDurationInMs,
category: ClapSegmentCategory.DIALOGUE,
prompt: speech,
outputType: ClapOutputType.AUDIO,
status: ClapSegmentStatus.TO_GENERATE,
}))
// the presence of a camera is mandatory
clap.segments.push(newSegment({
track: 4,
startTimeInMs: currentElapsedTimeInMs,
assetDurationInMs: defaultSegmentDurationInMs,
category: ClapSegmentCategory.CAMERA,
prompt: "movie still",
outputType: ClapOutputType.TEXT,
status: ClapSegmentStatus.COMPLETED,
}))
currentElapsedTimeInMs += defaultSegmentDurationInMs
}
set({ currentClap: clap })
return clap
},
convertClapToComic: async (clap: ClapProject): Promise<{
currentNbPanels: number
prompt: string
preset: Preset
layout: LayoutName
storyPrompt: string
stylePrompt: string
panels: string[]
renderedScenes: Record<string, RenderedScene>
speeches: string[]
captions: string[]
}> => {
const prompt = clap.meta.description
const [stylePrompt, storyPrompt] = prompt.split("||").map(x => x.trim())
const panels: string[] = []
const renderedScenes: Record<string, RenderedScene> = {}
const captions: string[] = []
const speeches: string[] = []
const panelGenerationStatus: Record<number, boolean> = {}
const cameraShots = clap.segments.filter(s => s.category === ClapSegmentCategory.CAMERA)
const shots = cameraShots.map(cameraShot => ({
camera: cameraShot,
storyboard: filterSegments(
ClapSegmentFilteringMode.START,
cameraShot,
clap.segments,
ClapSegmentCategory.STORYBOARD,
).at(0) as (ClapSegment | undefined),
ui: filterSegments(
ClapSegmentFilteringMode.START,
cameraShot,
clap.segments,
ClapSegmentCategory.INTERFACE,
).at(0) as (ClapSegment | undefined),
dialogue: filterSegments(
ClapSegmentFilteringMode.START,
cameraShot,
clap.segments,
ClapSegmentCategory.DIALOGUE,
).at(0) as (ClapSegment | undefined)
})).filter(item => item.storyboard && item.ui) as {
camera: ClapSegment
storyboard: ClapSegment
ui: ClapSegment
dialogue: ClapSegment
}[]
shots.forEach(({ camera, storyboard, ui, dialogue }, id) => {
panels.push(storyboard.prompt)
const renderedScene: RenderedScene = {
renderId: storyboard?.id || "",
status: "pending",
assetUrl: "",
alt: storyboard?.prompt || "",
error: "",
maskUrl: "",
segments: []
}
if (storyboard?.assetUrl) {
renderedScene.assetUrl = storyboard.assetUrl
renderedScene.status = "pregenerated" // <- special trick to indicate that it should not be re-generated
}
renderedScenes[id] = renderedScene
panelGenerationStatus[id] = false
speeches.push(dialogue?.prompt || "")
captions.push(ui?.prompt || "")
})
return {
currentNbPanels: shots.length,
prompt,
preset: parsePresetFromPrompts(panels),
layout: await parseLayoutFromStoryboards(shots.map(x => x.storyboard)),
storyPrompt,
stylePrompt,
panels,
renderedScenes,
speeches,
captions,
}
},
loadClap: async (blob: Blob) => {
const { convertClapToComic, currentNbPanelsPerPage } = get()
const currentClap = await parseClap(blob)
const {
currentNbPanels,
prompt,
preset,
layout,
storyPrompt,
stylePrompt,
panels,
renderedScenes,
speeches,
captions,
} = await convertClapToComic(currentClap)
// kids, don't do this in your projects: use state managers instead!
putTextInInput(document.getElementById("top-menu-input-style-prompt") as HTMLInputElement, stylePrompt)
putTextInInput(document.getElementById("top-menu-input-story-prompt") as HTMLInputElement, storyPrompt)
set({
currentClap,
currentNbPanels,
prompt,
preset,
// layout,
panels,
renderedScenes,
speeches,
captions,
currentNbPages: Math.round(currentNbPanels / currentNbPanelsPerPage),
upscaleQueue: {},
isGeneratingStory: false,
isGeneratingText: false,
})
},
downloadClap: async () => {
const { convertComicToClap, prompt } = get()
const currentClap = await convertComicToClap()
if (!currentClap) { throw new Error(`cannot save a clap.. if there is no clap`) }
const currentClapBlob: Blob = await serializeClap(currentClap)
// Create an object URL for the compressed clap blob
const objectUrl = URL.createObjectURL(currentClapBlob)
// Create an anchor element and force browser download
const anchor = document.createElement("a")
anchor.href = objectUrl
const [stylePrompt, storyPrompt] = prompt.split("||").map(x => x.trim())
const cleanStylePrompt = (stylePrompt || "").replace(/([^a-z0-9, ]+)/gi, " ")
const firstPartOfStory = (storyPrompt || "").split(",").shift() || ""
const cleanStoryPrompt = firstPartOfStory.replace(/([^a-z0-9, ]+)/gi, " ")
const cleanName = `${cleanStoryPrompt.slice(0, 90)} (${cleanStylePrompt.slice(0, 90) || "default style"})`
anchor.download = `${cleanName}.clap`
document.body.appendChild(anchor) // Append to the body (could be removed once clicked)
anchor.click() // Trigger the download
// Cleanup: revoke the object URL and remove the anchor element
URL.revokeObjectURL(objectUrl)
document.body.removeChild(anchor)
},
}))