Spaces:
Runtime error
Runtime error
first
Browse files- frontend/src/app.d.ts +6 -2
- frontend/src/lib/App.svelte +136 -66
- frontend/src/lib/ChatInput.svelte +69 -19
- frontend/src/lib/ChatMessage.svelte +17 -15
- frontend/src/lib/Icons/IconDelete.svelte +21 -0
- frontend/src/lib/Input +0 -0
- frontend/src/lib/initImage.ts +2 -0
- frontend/src/lib/store.ts +20 -39
- frontend/src/lib/types.ts +13 -1
- frontend/src/lib/utils.ts +22 -0
- frontend/static/statue.jpg +0 -0
frontend/src/app.d.ts
CHANGED
@@ -6,7 +6,11 @@ declare global {
|
|
6 |
// interface Locals {}
|
7 |
// interface PageData {}
|
8 |
// interface Platform {}
|
|
|
|
|
|
|
|
|
|
|
9 |
}
|
10 |
}
|
11 |
-
|
12 |
-
export {};
|
|
|
6 |
// interface Locals {}
|
7 |
// interface PageData {}
|
8 |
// interface Platform {}
|
9 |
+
interface Window {
|
10 |
+
parentIFrame: {
|
11 |
+
scrollTo: (x: number, y: number) => void;
|
12 |
+
};
|
13 |
+
}
|
14 |
}
|
15 |
}
|
16 |
+
export { };
|
|
frontend/src/lib/App.svelte
CHANGED
@@ -1,29 +1,44 @@
|
|
1 |
<script lang="ts">
|
2 |
import { PUBLIC_BACKEND_WS_URL } from '$env/static/public';
|
3 |
-
import { onMount } from 'svelte';
|
4 |
import { nanoid } from 'nanoid';
|
5 |
-
import { chatsStore, selectedChatId, loadingState } from '$lib/store';
|
6 |
-
import type { Message, Chat } from '$lib/types';
|
7 |
import { MessageType, Sender } from '$lib/types';
|
|
|
8 |
import ChatInput from '$lib/ChatInput.svelte';
|
9 |
import ChatMessage from '$lib/ChatMessage.svelte';
|
10 |
import ChatNewBtn from '$lib/ChatNewBtn.svelte';
|
|
|
11 |
|
12 |
-
$: isLoading =
|
13 |
|
|
|
|
|
14 |
function clearStateMsg(t = 5000) {
|
15 |
setTimeout(() => {
|
16 |
$loadingState = '';
|
17 |
}, t);
|
18 |
}
|
19 |
onMount(() => {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
20 |
// generateImage();
|
21 |
});
|
22 |
|
23 |
$: chatData = $chatsStore.find((chat) => chat.id === $selectedChatId);
|
24 |
-
$: messages = chatData?.messages || [];
|
25 |
-
|
26 |
-
|
|
|
|
|
|
|
|
|
27 |
|
28 |
function newChat() {
|
29 |
const chatId = nanoid();
|
@@ -36,16 +51,45 @@
|
|
36 |
$chatsStore = [chat].concat($chatsStore);
|
37 |
$selectedChatId = chat.id;
|
38 |
}
|
|
|
|
|
|
|
39 |
|
40 |
function submitMessage(event: CustomEvent) {
|
41 |
-
|
|
|
|
|
|
|
42 |
const message: Message = {
|
43 |
sender: Sender.USER,
|
44 |
id: nanoid(),
|
45 |
-
type:
|
46 |
-
content:
|
47 |
timestamp: new Date().getTime()
|
48 |
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
49 |
$chatsStore = $chatsStore.map((chat) => {
|
50 |
if (chat.id === $selectedChatId) {
|
51 |
chat.messages.push(message);
|
@@ -53,14 +97,11 @@
|
|
53 |
return chat;
|
54 |
});
|
55 |
}
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
}).format;
|
62 |
-
async function generateImage() {
|
63 |
-
if (isLoading) return;
|
64 |
|
65 |
$loadingState = 'Pending';
|
66 |
|
@@ -69,10 +110,9 @@
|
|
69 |
fn_index: 1,
|
70 |
session_hash: sessionHash
|
71 |
};
|
72 |
-
const image = '';
|
73 |
const datapayload = {
|
74 |
data: [
|
75 |
-
|
76 |
10.5, // text guidance
|
77 |
1.5, // image guidance
|
78 |
image,
|
@@ -83,8 +123,7 @@
|
|
83 |
0 // seed
|
84 |
]
|
85 |
};
|
86 |
-
|
87 |
-
const websocket = new WebSocket(PUBLIC_BACKEND_WS_URL);
|
88 |
// websocket.onopen = async function (event) {
|
89 |
// websocket.send(JSON.stringify({ hash: sessionHash }));
|
90 |
// };
|
@@ -119,25 +158,37 @@
|
|
119 |
break;
|
120 |
case 'process_completed':
|
121 |
try {
|
122 |
-
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
136 |
-
|
137 |
-
|
138 |
-
|
139 |
-
|
140 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
141 |
$loadingState = data.success ? 'Complete' : 'Error';
|
142 |
clearStateMsg();
|
143 |
} catch (err) {
|
@@ -163,38 +214,57 @@
|
|
163 |
<div>
|
164 |
<h1 class="text-2xl">CHATS</h1>
|
165 |
<div class="grid min-h-[40rem] grid-cols-4">
|
166 |
-
<div class="col-span-1 flex flex-col border-r p-4">
|
167 |
-
<
|
168 |
-
|
169 |
-
|
170 |
-
{#
|
171 |
-
|
172 |
-
<div
|
173 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
174 |
{chat.id === $selectedChatId ? 'bg-gray-400' : ''}"
|
175 |
-
|
176 |
-
|
177 |
-
|
178 |
-
|
179 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
180 |
</div>
|
181 |
-
|
182 |
-
{
|
183 |
-
|
184 |
-
|
185 |
-
|
186 |
-
|
187 |
-
|
188 |
-
|
189 |
-
|
190 |
-
|
191 |
</div>
|
192 |
</div>
|
193 |
-
|
|
|
194 |
{#each messages as message}
|
195 |
<ChatMessage {message} />
|
196 |
{/each}
|
197 |
-
<ChatInput on:submitMessage={submitMessage} />
|
|
|
|
|
|
|
198 |
</div>
|
199 |
</div>
|
200 |
</div>
|
|
|
1 |
<script lang="ts">
|
2 |
import { PUBLIC_BACKEND_WS_URL } from '$env/static/public';
|
3 |
+
import { onMount, tick } from 'svelte';
|
4 |
import { nanoid } from 'nanoid';
|
5 |
+
import { chatsStore, selectedChatId, loadingState, lastBase64Image } from '$lib/store';
|
6 |
+
import type { Message, Chat, InferenceResponse, ImageFile } from '$lib/types';
|
7 |
import { MessageType, Sender } from '$lib/types';
|
8 |
+
import { timeFormater, fetchImageBase64 } from '$lib/utils';
|
9 |
import ChatInput from '$lib/ChatInput.svelte';
|
10 |
import ChatMessage from '$lib/ChatMessage.svelte';
|
11 |
import ChatNewBtn from '$lib/ChatNewBtn.svelte';
|
12 |
+
import IconDelete from './Icons/IconDelete.svelte';
|
13 |
|
14 |
+
$: isLoading = !($loadingState === '' || $loadingState === 'Complete');
|
15 |
|
16 |
+
let chatBoxEl: HTMLDivElement;
|
17 |
+
let chatInputEl: HTMLInputElement;
|
18 |
function clearStateMsg(t = 5000) {
|
19 |
setTimeout(() => {
|
20 |
$loadingState = '';
|
21 |
}, t);
|
22 |
}
|
23 |
onMount(() => {
|
24 |
+
const observer = new ResizeObserver(() => {
|
25 |
+
window.scrollTo(0, chatBoxEl.getBoundingClientRect().height);
|
26 |
+
if ('parentIFrame' in window) {
|
27 |
+
(window as any).parentIFrame.scrollTo(0, chatBoxEl.getBoundingClientRect().height);
|
28 |
+
}
|
29 |
+
});
|
30 |
+
observer.observe(chatBoxEl);
|
31 |
// generateImage();
|
32 |
});
|
33 |
|
34 |
$: chatData = $chatsStore.find((chat) => chat.id === $selectedChatId);
|
35 |
+
$: messages = chatData?.messages.sort((a, b) => a.timestamp - b.timestamp) || [];
|
36 |
+
$: if ($selectedChatId) {
|
37 |
+
// find last message with image type
|
38 |
+
$lastBase64Image =
|
39 |
+
[...messages].reverse().find((message) => message.type === MessageType.IMAGE)?.content ||
|
40 |
+
null;
|
41 |
+
}
|
42 |
|
43 |
function newChat() {
|
44 |
const chatId = nanoid();
|
|
|
51 |
$chatsStore = [chat].concat($chatsStore);
|
52 |
$selectedChatId = chat.id;
|
53 |
}
|
54 |
+
function deleteChat(id: string) {
|
55 |
+
$chatsStore = $chatsStore.filter((chat) => chat.id !== id);
|
56 |
+
}
|
57 |
|
58 |
function submitMessage(event: CustomEvent) {
|
59 |
+
if ($chatsStore.length === 0) {
|
60 |
+
newChat();
|
61 |
+
}
|
62 |
+
const { type, content } = event.detail;
|
63 |
const message: Message = {
|
64 |
sender: Sender.USER,
|
65 |
id: nanoid(),
|
66 |
+
type: type,
|
67 |
+
content: content,
|
68 |
timestamp: new Date().getTime()
|
69 |
};
|
70 |
+
updateChatStore(message);
|
71 |
+
|
72 |
+
if (type === MessageType.IMAGE) {
|
73 |
+
// upate last image to be the last image sent
|
74 |
+
$lastBase64Image = content;
|
75 |
+
} else if (type === MessageType.TEXT) {
|
76 |
+
// if the last message was an image, then we want to run inference
|
77 |
+
// on the image and the text
|
78 |
+
if (!$lastBase64Image) {
|
79 |
+
const message: Message = {
|
80 |
+
sender: Sender.BOT,
|
81 |
+
id: nanoid(),
|
82 |
+
type: MessageType.TEXT,
|
83 |
+
content: "Sorry, I don't have an image to work with.",
|
84 |
+
timestamp: new Date().getTime()
|
85 |
+
};
|
86 |
+
updateChatStore(message);
|
87 |
+
return;
|
88 |
+
}
|
89 |
+
runInference($lastBase64Image, content);
|
90 |
+
}
|
91 |
+
}
|
92 |
+
function updateChatStore(message: Message) {
|
93 |
$chatsStore = $chatsStore.map((chat) => {
|
94 |
if (chat.id === $selectedChatId) {
|
95 |
chat.messages.push(message);
|
|
|
97 |
return chat;
|
98 |
});
|
99 |
}
|
100 |
+
|
101 |
+
async function runInference(image: string, prompt: string) {
|
102 |
+
if (isLoading || image === '' || prompt === '') {
|
103 |
+
return;
|
104 |
+
}
|
|
|
|
|
|
|
105 |
|
106 |
$loadingState = 'Pending';
|
107 |
|
|
|
110 |
fn_index: 1,
|
111 |
session_hash: sessionHash
|
112 |
};
|
|
|
113 |
const datapayload = {
|
114 |
data: [
|
115 |
+
prompt, // prompt
|
116 |
10.5, // text guidance
|
117 |
1.5, // image guidance
|
118 |
image,
|
|
|
123 |
0 // seed
|
124 |
]
|
125 |
};
|
126 |
+
const websocket = new WebSocket(`wss://${PUBLIC_BACKEND_WS_URL}/queue/join`);
|
|
|
127 |
// websocket.onopen = async function (event) {
|
128 |
// websocket.send(JSON.stringify({ hash: sessionHash }));
|
129 |
// };
|
|
|
158 |
break;
|
159 |
case 'process_completed':
|
160 |
try {
|
161 |
+
const response = data.output as InferenceResponse;
|
162 |
+
const imageData = response.data[0] as ImageFile[];
|
163 |
+
const nsfwData = data.output.data[1] as boolean[] | null;
|
164 |
+
console.log('imageData', imageData);
|
165 |
+
console.log('nsfwData', nsfwData);
|
166 |
+
|
167 |
+
if (nsfwData && nsfwData[0]) {
|
168 |
+
const message: Message = {
|
169 |
+
sender: Sender.BOT,
|
170 |
+
id: nanoid(),
|
171 |
+
type: MessageType.TEXT,
|
172 |
+
content:
|
173 |
+
'Sorry this prompt possibly generates NSFW content. Please try another prompt.',
|
174 |
+
timestamp: new Date().getTime()
|
175 |
+
};
|
176 |
+
updateChatStore(message);
|
177 |
+
} else {
|
178 |
+
const fileName = imageData[0].name;
|
179 |
+
const imageBase64 = await fetchImageBase64(
|
180 |
+
`https://${PUBLIC_BACKEND_WS_URL}/file=${fileName}`
|
181 |
+
);
|
182 |
+
const message: Message = {
|
183 |
+
sender: Sender.BOT,
|
184 |
+
id: nanoid(),
|
185 |
+
type: MessageType.IMAGE,
|
186 |
+
content: imageBase64,
|
187 |
+
timestamp: new Date().getTime()
|
188 |
+
};
|
189 |
+
$lastBase64Image = imageBase64;
|
190 |
+
updateChatStore(message);
|
191 |
+
}
|
192 |
$loadingState = data.success ? 'Complete' : 'Error';
|
193 |
clearStateMsg();
|
194 |
} catch (err) {
|
|
|
214 |
<div>
|
215 |
<h1 class="text-2xl">CHATS</h1>
|
216 |
<div class="grid min-h-[40rem] grid-cols-4">
|
217 |
+
<div class="col-span-1 flex flex-col border-r p-4 relative">
|
218 |
+
<div class="sticky top-3">
|
219 |
+
<ChatNewBtn on:click={newChat} />
|
220 |
+
<div class="max-h-[40rem] flex flex-col gap-2 overflow-y-scroll">
|
221 |
+
{#if $chatsStore.length}
|
222 |
+
{#each $chatsStore as chat}
|
223 |
+
<div class="flex flex-col relative">
|
224 |
+
<button
|
225 |
+
class="disabled:opacity-60 disabled:cursor-progress"
|
226 |
+
on:click={() => ($selectedChatId = chat.id)}
|
227 |
+
disabled={isLoading}
|
228 |
+
>
|
229 |
+
<div
|
230 |
+
class=" flex flex-col h-16 items-start justify-center rounded-xl bg-gray-100 px-4 text-gray-900
|
231 |
{chat.id === $selectedChatId ? 'bg-gray-400' : ''}"
|
232 |
+
>
|
233 |
+
<h3 class="w-full truncate font-semibold">{chat.blurb}</h3>
|
234 |
+
<p class="w-full truncate text-sm text-gray-500">
|
235 |
+
{timeFormater(new Date(chat.timestamp))}
|
236 |
+
</p>
|
237 |
+
</div>
|
238 |
+
</button>
|
239 |
+
<button
|
240 |
+
class="text-black absolute right-1 bottom-1 disabled:opacity-60"
|
241 |
+
on:click={() => deleteChat(chat.id)}
|
242 |
+
disabled={isLoading}
|
243 |
+
>
|
244 |
+
<IconDelete />
|
245 |
+
</button>
|
246 |
</div>
|
247 |
+
{/each}
|
248 |
+
{:else}
|
249 |
+
<div
|
250 |
+
class="flex h-16 flex-col items-start justify-center rounded-xl bg-gray-100 px-4 text-gray-900"
|
251 |
+
>
|
252 |
+
<h3 class="w-full truncate font-semibold">No chats</h3>
|
253 |
+
<p class="w-full truncate text-sm text-gray-500">Start a new Chat!</p>
|
254 |
+
</div>
|
255 |
+
{/if}
|
256 |
+
</div>
|
257 |
</div>
|
258 |
</div>
|
259 |
+
|
260 |
+
<div class="col-span-3 flex flex-col" bind:this={chatBoxEl}>
|
261 |
{#each messages as message}
|
262 |
<ChatMessage {message} />
|
263 |
{/each}
|
264 |
+
<ChatInput on:submitMessage={submitMessage} bind:inputEl={chatInputEl} disabled={isLoading} />
|
265 |
+
</div>
|
266 |
+
<div class="top-0 right-0 z-10">
|
267 |
+
Loading: {$loadingState}
|
268 |
</div>
|
269 |
</div>
|
270 |
</div>
|
frontend/src/lib/ChatInput.svelte
CHANGED
@@ -1,39 +1,89 @@
|
|
1 |
<script lang="ts">
|
2 |
import { createEventDispatcher, onMount } from 'svelte';
|
|
|
|
|
|
|
|
|
3 |
|
4 |
const dispatch = createEventDispatcher();
|
5 |
|
6 |
let textInput: string = '';
|
7 |
-
let
|
8 |
|
9 |
function onSubmit(event: Event) {
|
10 |
event.stopPropagation();
|
11 |
event.preventDefault();
|
12 |
event.stopImmediatePropagation();
|
13 |
-
|
14 |
if (textInput.trim() !== '') {
|
15 |
-
dispatch('submitMessage',
|
|
|
|
|
|
|
16 |
textInput = '';
|
17 |
}
|
18 |
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
19 |
onMount(() => {
|
20 |
inputEl.focus();
|
|
|
|
|
|
|
21 |
});
|
22 |
</script>
|
23 |
|
24 |
-
<
|
25 |
-
<
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
<script lang="ts">
|
2 |
import { createEventDispatcher, onMount } from 'svelte';
|
3 |
+
import { MessageType } from '$lib/types';
|
4 |
+
|
5 |
+
export let inputEl: HTMLInputElement;
|
6 |
+
export let disabled: boolean = true;
|
7 |
|
8 |
const dispatch = createEventDispatcher();
|
9 |
|
10 |
let textInput: string = '';
|
11 |
+
let isDragOver: boolean = false;
|
12 |
|
13 |
function onSubmit(event: Event) {
|
14 |
event.stopPropagation();
|
15 |
event.preventDefault();
|
16 |
event.stopImmediatePropagation();
|
|
|
17 |
if (textInput.trim() !== '') {
|
18 |
+
dispatch('submitMessage', {
|
19 |
+
type: MessageType.TEXT,
|
20 |
+
content: textInput
|
21 |
+
});
|
22 |
textInput = '';
|
23 |
}
|
24 |
}
|
25 |
+
function onDrop(event: DragEvent) {
|
26 |
+
event.stopPropagation();
|
27 |
+
event.preventDefault();
|
28 |
+
event.stopImmediatePropagation();
|
29 |
+
if (event && event.dataTransfer) {
|
30 |
+
const files = [...event.dataTransfer.files];
|
31 |
+
if (files.length) {
|
32 |
+
const file = files[0];
|
33 |
+
if (file.type.startsWith('image/') && file.size < 1.5e7) {
|
34 |
+
const reader = new FileReader();
|
35 |
+
reader.readAsDataURL(file);
|
36 |
+
reader.onload = () => {
|
37 |
+
dispatch('submitMessage', {
|
38 |
+
type: MessageType.IMAGE,
|
39 |
+
content: reader.result
|
40 |
+
});
|
41 |
+
};
|
42 |
+
}
|
43 |
+
}
|
44 |
+
}
|
45 |
+
isDragOver = false;
|
46 |
+
}
|
47 |
+
|
48 |
+
function onDragover(event: DragEvent) {
|
49 |
+
event.stopPropagation();
|
50 |
+
event.preventDefault();
|
51 |
+
if (event && event.dataTransfer) {
|
52 |
+
event.dataTransfer.dropEffect = 'copy';
|
53 |
+
}
|
54 |
+
isDragOver = true;
|
55 |
+
}
|
56 |
+
|
57 |
onMount(() => {
|
58 |
inputEl.focus();
|
59 |
+
window.addEventListener('dragover', onDragover, false);
|
60 |
+
window.addEventListener('drop', onDrop, false);
|
61 |
+
window.addEventListener('dragleave', () => (isDragOver = false), false);
|
62 |
});
|
63 |
</script>
|
64 |
|
65 |
+
<div class="mt-auto {isDragOver ? 'bg-yellow-50' : ''} py-4">
|
66 |
+
<form class="flex selection:w-full gap-2 border-t px-3 pt-4 pb-6" on:submit={onSubmit}>
|
67 |
+
<input
|
68 |
+
bind:value={textInput}
|
69 |
+
bind:this={inputEl}
|
70 |
+
on:click|stopPropagation
|
71 |
+
{disabled}
|
72 |
+
type="text"
|
73 |
+
class="flex h-11 flex-1 items-center rounded-full border bg-gray-100 px-4 text-black disabled:opacity-60 disabled:cursor-progress"
|
74 |
+
placeholder="Type here"
|
75 |
+
title="Type here to send a message"
|
76 |
+
/>
|
77 |
+
<button
|
78 |
+
{disabled}
|
79 |
+
title="Send your message"
|
80 |
+
type="submit"
|
81 |
+
class="rounded-full bg-black dark:bg-white dark:text-black px-4 font-semibold text-gray-200 disabled:opacity-60 disabled:cursor-progress"
|
82 |
+
>Submit</button
|
83 |
+
>
|
84 |
+
</form>
|
85 |
+
<div class="flex items-center px-3 pt-4 pb-6 bg-yellow-50 py-3 rounded">
|
86 |
+
<i class="fas fa-image text-2xl text-gray-500 mr-2" />
|
87 |
+
<h2 class="text-xl font-bold text-gray-500">Drop your image here to start</h2>
|
88 |
+
</div>
|
89 |
+
</div>
|
frontend/src/lib/ChatMessage.svelte
CHANGED
@@ -3,22 +3,24 @@
|
|
3 |
import { MessageType, Sender } from '$lib/types';
|
4 |
|
5 |
export let message: Message;
|
|
|
6 |
</script>
|
7 |
|
8 |
<div class="flex flex-col gap-4 p-4">
|
9 |
-
<
|
10 |
-
|
11 |
-
|
12 |
-
|
13 |
-
|
14 |
-
|
15 |
-
|
16 |
-
|
17 |
-
|
18 |
-
|
19 |
-
|
20 |
-
|
21 |
-
|
22 |
-
|
23 |
-
|
|
|
24 |
</div>
|
|
|
3 |
import { MessageType, Sender } from '$lib/types';
|
4 |
|
5 |
export let message: Message;
|
6 |
+
export let loading: boolean = false;
|
7 |
</script>
|
8 |
|
9 |
<div class="flex flex-col gap-4 p-4">
|
10 |
+
<div class={message.sender === Sender.USER ? 'self-end' : 'self-start'}>
|
11 |
+
{#if loading}
|
12 |
+
<div class="flex flex-col gap-2">
|
13 |
+
<div class="w-32 h-8 bg-gray-200 rounded-xl animate-pulse" />
|
14 |
+
<div class="w-32 h-8 bg-gray-200 rounded-xl animate-pulse" />
|
15 |
+
</div>
|
16 |
+
{:else if message.type === MessageType.TEXT}
|
17 |
+
<p
|
18 |
+
class="self-end rounded-3 -mt-2 rounded-t rounded-b-2xl bg-black dark:bg-white py-2 px-4 text-white dark:text-black"
|
19 |
+
>
|
20 |
+
{message.content}
|
21 |
+
</p>
|
22 |
+
{:else if message.type === MessageType.IMAGE}
|
23 |
+
<img class="max-h-[328px] self-start rounded-xl bg-gray-200" src={message.content} alt="" />
|
24 |
+
{/if}
|
25 |
+
</div>
|
26 |
</div>
|
frontend/src/lib/Icons/IconDelete.svelte
ADDED
@@ -0,0 +1,21 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<script lang="ts">
|
2 |
+
export let classNames = "";
|
3 |
+
</script>
|
4 |
+
|
5 |
+
<svg
|
6 |
+
class={classNames}
|
7 |
+
xmlns="http://www.w3.org/2000/svg"
|
8 |
+
xmlns:xlink="http://www.w3.org/1999/xlink"
|
9 |
+
aria-hidden="true"
|
10 |
+
focusable="false"
|
11 |
+
role="img"
|
12 |
+
width="1em"
|
13 |
+
height="1em"
|
14 |
+
preserveAspectRatio="xMidYMid meet"
|
15 |
+
viewBox="0 0 32 32"
|
16 |
+
>
|
17 |
+
<path d="M12 12h2v12h-2z" fill="currentColor" /><path d="M18 12h2v12h-2z" fill="currentColor" /><path
|
18 |
+
d="M4 6v2h2v20a2 2 0 0 0 2 2h16a2 2 0 0 0 2-2V8h2V6zm4 22V8h16v20z"
|
19 |
+
fill="currentColor"
|
20 |
+
/><path d="M12 2h8v2h-8z" fill="currentColor" />
|
21 |
+
</svg>
|
frontend/src/lib/Input
DELETED
File without changes
|
frontend/src/lib/initImage.ts
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
export const initImage =
|
2 |
+
"";
|
frontend/src/lib/store.ts
CHANGED
@@ -1,17 +1,25 @@
|
|
1 |
import localforage from 'localforage';
|
2 |
import { writable } from 'svelte/store';
|
3 |
-
import type
|
4 |
import { nanoid } from 'nanoid';
|
|
|
5 |
|
6 |
const intitalChatId = nanoid();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
7 |
const initialData: ChatData = [{
|
8 |
id: intitalChatId,
|
9 |
-
messages: [],
|
10 |
blurb: `New Chat - ${intitalChatId}`,
|
11 |
timestamp: new Date().getTime()
|
12 |
}];
|
13 |
|
14 |
-
|
15 |
const loadingState = writable<string>('');
|
16 |
const chatsStore = writable<ChatData>(initialData);
|
17 |
const selectedChatId = writable<string>(intitalChatId);
|
@@ -29,7 +37,10 @@ localforage.getItem<ChatData>('chatsStore').then((value) => {
|
|
29 |
}
|
30 |
});
|
31 |
|
32 |
-
chatsStore.subscribe((value) =>
|
|
|
|
|
|
|
33 |
|
34 |
localforage.getItem<string>('selectedChatId').then((value) => {
|
35 |
if (value) {
|
@@ -39,39 +50,9 @@ localforage.getItem<string>('selectedChatId').then((value) => {
|
|
39 |
}
|
40 |
});
|
41 |
|
42 |
-
selectedChatId.subscribe((value) =>
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
// import { writable } from 'svelte/store';
|
47 |
-
// import { browser } from '$app/environment';
|
48 |
-
// import type { ChatData } from './types';
|
49 |
-
// import { nanoid } from 'nanoid';
|
50 |
-
|
51 |
-
|
52 |
-
// const intitalChatId = nanoid();
|
53 |
-
// const initialData: ChatData = [{
|
54 |
-
// id: intitalChatId,
|
55 |
-
// messages: [],
|
56 |
-
// blurb: `New Chat - ${new Date().getTime()}`
|
57 |
-
// }
|
58 |
-
// ]
|
59 |
-
|
60 |
-
// export const loadingState = writable<string>('');
|
61 |
-
// export const chatsStore = writable<ChatData>(
|
62 |
-
// browser ? JSON.parse(localStorage['chatsStore'] || JSON.stringify(initialData)) : initialData
|
63 |
-
// );
|
64 |
-
// chatsStore.subscribe((value) => {
|
65 |
-
// if (browser) {
|
66 |
-
// return (localStorage['chatsStore'] = JSON.stringify(value));
|
67 |
-
// }
|
68 |
-
// });
|
69 |
|
70 |
-
|
71 |
-
// browser ? JSON.parse(localStorage['selectedChatId'] || JSON.stringify(intitalChatId)) : intitalChatId
|
72 |
-
// );
|
73 |
-
// selectedChatId.subscribe((value) => {
|
74 |
-
// if (browser) {
|
75 |
-
// return (localStorage['selectedChatId'] = JSON.stringify(value));
|
76 |
-
// }
|
77 |
-
// });
|
|
|
1 |
import localforage from 'localforage';
|
2 |
import { writable } from 'svelte/store';
|
3 |
+
import { MessageType, Sender, type ChatData, type Message } from './types';
|
4 |
import { nanoid } from 'nanoid';
|
5 |
+
import { initImage } from './initImage';
|
6 |
|
7 |
const intitalChatId = nanoid();
|
8 |
+
const initMessage: Message = {
|
9 |
+
id: nanoid(),
|
10 |
+
content: initImage,
|
11 |
+
sender: Sender.USER,
|
12 |
+
type: MessageType.IMAGE,
|
13 |
+
timestamp: new Date().getTime()
|
14 |
+
}
|
15 |
const initialData: ChatData = [{
|
16 |
id: intitalChatId,
|
17 |
+
messages: [initMessage],
|
18 |
blurb: `New Chat - ${intitalChatId}`,
|
19 |
timestamp: new Date().getTime()
|
20 |
}];
|
21 |
|
22 |
+
const lastBase64Image = writable<string | null>(initMessage.content);
|
23 |
const loadingState = writable<string>('');
|
24 |
const chatsStore = writable<ChatData>(initialData);
|
25 |
const selectedChatId = writable<string>(intitalChatId);
|
|
|
37 |
}
|
38 |
});
|
39 |
|
40 |
+
chatsStore.subscribe((value) => {
|
41 |
+
localforage.setItem<ChatData>('chatsStore', value)
|
42 |
+
return value;
|
43 |
+
});
|
44 |
|
45 |
localforage.getItem<string>('selectedChatId').then((value) => {
|
46 |
if (value) {
|
|
|
50 |
}
|
51 |
});
|
52 |
|
53 |
+
selectedChatId.subscribe((value) => {
|
54 |
+
localforage.setItem<string>('selectedChatId', value)
|
55 |
+
return value;
|
56 |
+
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
57 |
|
58 |
+
export { loadingState, chatsStore, selectedChatId, lastBase64Image };
|
|
|
|
|
|
|
|
|
|
|
|
|
|
frontend/src/lib/types.ts
CHANGED
@@ -26,4 +26,16 @@ export interface Chat {
|
|
26 |
timestamp: number;
|
27 |
}
|
28 |
|
29 |
-
export type ChatData = Chat[];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
26 |
timestamp: number;
|
27 |
}
|
28 |
|
29 |
+
export type ChatData = Chat[];
|
30 |
+
|
31 |
+
export type ImageFile = {
|
32 |
+
name: string;
|
33 |
+
data: null | string;
|
34 |
+
is_file: boolean;
|
35 |
+
}
|
36 |
+
export interface InferenceResponse {
|
37 |
+
data: Array<Array<ImageFile | boolean> | null | number>;
|
38 |
+
is_generating: boolean;
|
39 |
+
duration: number;
|
40 |
+
average_duration: number;
|
41 |
+
}
|
frontend/src/lib/utils.ts
ADDED
@@ -0,0 +1,22 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
function blobToBase64(blob: Blob): Promise<string> {
|
2 |
+
return new Promise((resolve) => {
|
3 |
+
const reader = new FileReader();
|
4 |
+
reader.onloadend = () => resolve(reader.result as string);
|
5 |
+
reader.readAsDataURL(blob);
|
6 |
+
});
|
7 |
+
}
|
8 |
+
|
9 |
+
export const timeFormater = new Intl.DateTimeFormat('en-US', {
|
10 |
+
day: 'numeric',
|
11 |
+
month: 'short',
|
12 |
+
hour: 'numeric',
|
13 |
+
minute: 'numeric'
|
14 |
+
}).format;
|
15 |
+
|
16 |
+
export async function fetchImageBase64(url: string): Promise<string> {
|
17 |
+
const response = await fetch(url);
|
18 |
+
const blob = await response.blob();
|
19 |
+
const base64 = await blobToBase64(blob);
|
20 |
+
return base64;
|
21 |
+
}
|
22 |
+
|
frontend/static/statue.jpg
ADDED
![]() |