Spaces:
Runtime error
Runtime error
<script lang="ts"> | |
import Cursor from '$lib/Cursor.svelte'; | |
import Frame from '$lib/Frame.svelte'; | |
import PaintFrame from '$lib/PaintFrame.svelte'; | |
import PaintCanvas from '$lib/PaintCanvas.svelte'; | |
import ShareWithCommunity from '$lib/Buttons/ShareWithCommunity.svelte'; | |
import Menu from '$lib/Menu.svelte'; | |
import PromptModal from '$lib/PromptModal.svelte'; | |
import { COLORS } from '$lib/constants'; | |
import { PUBLIC_WS_INPAINTING } from '$env/static/public'; | |
import type { PromptImgKey } from '$lib/types'; | |
import { Status } from '$lib/types'; | |
import { loadingState, currZoomTransform, maskEl, selectedRoomID } from '$lib/store'; | |
import { useMyPresence, useObject, useOthers } from '$lib/liveblocks'; | |
import { base64ToBlob, uploadImage } from '$lib/utils'; | |
import { nanoid } from 'nanoid'; | |
const myPresence = useMyPresence({ addToHistory: true }); | |
const others = useOthers(); | |
function getKey(position: { x: number; y: number }): PromptImgKey { | |
return `${position.x}_${position.y}`; | |
} | |
const promptImgStorage = useObject('promptImgStorage'); | |
let showModal = false; | |
$: isLoading = $myPresence?.status === Status.loading || false; | |
function onPrompt() { | |
if (!isLoading && !showModal) { | |
showModal = true; | |
myPresence.update({ | |
status: Status.prompting | |
}); | |
} | |
} | |
function onClose() { | |
showModal = false; | |
} | |
function onPaint() { | |
generateImage(); | |
showModal = false; | |
} | |
async function generateImage() { | |
if (isLoading) return; | |
$loadingState = 'Pending'; | |
const prompt = $myPresence.currentPrompt; | |
const position = $myPresence.frame; | |
const room = $selectedRoomID || 'default'; | |
console.log('Generating...', prompt, position); | |
myPresence.update({ | |
status: Status.loading | |
}); | |
const sessionHash = crypto.randomUUID(); | |
const base64Crop = $maskEl.toDataURL('image/png'); | |
const hashpayload = { | |
fn_index: 0, | |
session_hash: sessionHash | |
}; | |
const datapayload = { | |
data: [base64Crop, prompt, 0.75, 7.5, 40, 'patchmatch'] | |
}; | |
const websocket = new WebSocket(PUBLIC_WS_INPAINTING); | |
// websocket.onopen = async function (event) { | |
// websocket.send(JSON.stringify({ hash: sessionHash })); | |
// }; | |
websocket.onclose = (evt) => { | |
if (!evt.wasClean) { | |
$loadingState = 'Error'; | |
myPresence.update({ | |
status: Status.ready | |
}); | |
} | |
}; | |
websocket.onmessage = async function (event) { | |
try { | |
const data = JSON.parse(event.data); | |
$loadingState = ''; | |
switch (data.msg) { | |
case 'send_hash': | |
websocket.send(JSON.stringify(hashpayload)); | |
break; | |
case 'send_data': | |
$loadingState = 'Sending Data'; | |
websocket.send(JSON.stringify({ ...hashpayload, ...datapayload })); | |
break; | |
case 'queue_full': | |
$loadingState = 'Queue full'; | |
websocket.close(); | |
myPresence.update({ | |
status: Status.ready | |
}); | |
return; | |
case 'estimation': | |
const { rank, queue_size } = data; | |
$loadingState = `On queue ${rank}/${queue_size}`; | |
break; | |
case 'process_generating': | |
$loadingState = data.success ? 'Generating' : 'Error'; | |
break; | |
case 'process_completed': | |
try { | |
const params = data.output.data[0] as { | |
is_nsfw: boolean; | |
image: { | |
url: string; | |
filename: string; | |
}; | |
}; | |
const isNSWF = params.is_nsfw; | |
if (isNSWF) { | |
throw new Error('NFSW'); | |
} | |
const key = getKey(position); | |
// const imgBlob = await base64ToBlob(imgBase64); | |
const promptImgParams = { | |
prompt, | |
imgURL: params.image.filename, | |
position, | |
date: new Date().getTime(), | |
id: nanoid(), | |
room: room | |
}; | |
// const imgURL = await uploadImage(imgBlob, promptImgParams); | |
$promptImgStorage.set(key, promptImgParams); | |
console.log(params.image.url); | |
$loadingState = data.success ? 'Complete' : 'Error'; | |
setTimeout(() => { | |
$loadingState = ''; | |
}, 2000); | |
myPresence.update({ | |
status: Status.ready, | |
currentPrompt: '' | |
}); | |
} catch (err) { | |
const tError = err as Error; | |
$loadingState = tError?.message; | |
myPresence.update({ | |
status: Status.ready | |
}); | |
setTimeout(() => { | |
$loadingState = ''; | |
}, 10000); | |
} | |
websocket.close(); | |
return; | |
case 'process_starts': | |
$loadingState = 'Processing'; | |
break; | |
} | |
} catch (e) { | |
console.error(e); | |
$loadingState = 'Error'; | |
} | |
}; | |
} | |
</script> | |
<!-- Show the current user's cursor location --> | |
<div class="text touch-none pointer-events-none"> | |
{$loadingState} | |
</div> | |
{#if showModal} | |
<PromptModal on:paint={onPaint} on:close={onClose} initPrompt={$myPresence?.currentPrompt} /> | |
{/if} | |
<div class="fixed top-0 left-0 z-0 w-screen h-screen min-h-[600px]"> | |
<PaintCanvas /> | |
<main class="z-10 relative"> | |
<PaintFrame on:prompt={onPrompt} transform={$currZoomTransform} /> | |
<!-- When others connected, iterate through others and show their cursors --> | |
{#if $others} | |
{#each [...$others] as { connectionId, presence } (connectionId)} | |
{#if (presence?.status === Status.loading || presence?.status === Status.prompting || presence?.status === Status.masking) && presence?.frame} | |
<Frame | |
isLoading={presence?.status === Status.loading} | |
position={presence?.frame} | |
prompt={presence?.currentPrompt} | |
transform={$currZoomTransform} | |
/> | |
{/if} | |
{#if presence?.cursor} | |
<Cursor | |
color={COLORS[1 + (connectionId % (COLORS.length - 1))]} | |
position={presence?.cursor} | |
transform={$currZoomTransform} | |
/> | |
{/if} | |
{/each} | |
{/if} | |
</main> | |
</div> | |
<!-- <div class="fixed top-0 right-0 z-10 p-2"> | |
<ShareWithCommunity /> | |
</div> --> | |
<div class="fixed bottom-2 md:bottom-16 left-0 right-0 z-10 my-2"> | |
<Menu on:prompt={onPrompt} {isLoading} /> | |
</div> | |
<style lang="postcss" scoped> | |
</style> | |