radames commited on
Commit
18cd77b
·
1 Parent(s): 6724b65
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 = false;
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
- $: console.log($chatsStore);
 
 
 
 
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
- const value = event.detail;
 
 
 
42
  const message: Message = {
43
  sender: Sender.USER,
44
  id: nanoid(),
45
- type: MessageType.TEXT,
46
- content: value,
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
- const timeFormater = new Intl.DateTimeFormat('en-US', {
57
- day: 'numeric',
58
- month: 'short',
59
- hour: 'numeric',
60
- minute: 'numeric'
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
- 'make him wear shirts', // prompt
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
- console.log(data);
123
- // const params = data.output.data[0] as {
124
- // is_nsfw: boolean;
125
- // image: {
126
- // url: string;
127
- // filename: string;
128
- // };
129
- // };
130
- // const isNSWF = params.is_nsfw;
131
- // if (isNSWF) {
132
- // throw new Error('NFSW');
133
- // }
134
- // // const imgBlob = await base64ToBlob(imgBase64);
135
- // const promptImgParams = {
136
- // imgURL: params.image.filename
137
- // };
138
- // // const imgURL = await uploadImage(imgBlob, promptImgParams);
139
- // // $promptImgStorage.set(imageKey, promptImgParams);
140
- // console.log(params.image.url);
 
 
 
 
 
 
 
 
 
 
 
 
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
- <ChatNewBtn on:click={newChat} />
168
- <div class="max-h-[40rem] flex flex-col gap-2 overflow-y-scroll">
169
- {#if $chatsStore.length}
170
- {#each $chatsStore as chat}
171
- <button on:click={() => ($selectedChatId = chat.id)}>
172
- <div
173
- class="flex h-16 flex-col items-start justify-center rounded-xl bg-gray-100 px-4 text-gray-900
 
 
 
 
 
 
174
  {chat.id === $selectedChatId ? 'bg-gray-400' : ''}"
175
- >
176
- <h3 class="w-full truncate font-semibold">{chat.blurb}</h3>
177
- <p class="w-full truncate text-sm text-gray-500">
178
- {timeFormater(new Date(chat.timestamp))}
179
- </p>
 
 
 
 
 
 
 
 
 
180
  </div>
181
- </button>
182
- {/each}
183
- {:else}
184
- <div
185
- class="flex h-16 flex-col items-start justify-center rounded-xl bg-gray-100 px-4 text-gray-900"
186
- >
187
- <h3 class="w-full truncate font-semibold">No chats</h3>
188
- <p class="w-full truncate text-sm text-gray-500">Start a new Chat!</p>
189
- </div>
190
- {/if}
191
  </div>
192
  </div>
193
- <div class="col-span-3 flex flex-col">
 
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 inputEl: HTMLInputElement;
8
 
9
  function onSubmit(event: Event) {
10
  event.stopPropagation();
11
  event.preventDefault();
12
  event.stopImmediatePropagation();
13
-
14
  if (textInput.trim() !== '') {
15
- dispatch('submitMessage', textInput);
 
 
 
16
  textInput = '';
17
  }
18
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
19
  onMount(() => {
20
  inputEl.focus();
 
 
 
21
  });
22
  </script>
23
 
24
- <form class="mt-auto flex w-full gap-2 border-t px-3 pt-4 pb-6" on:submit={onSubmit}>
25
- <input
26
- bind:value={textInput}
27
- bind:this={inputEl}
28
- on:click|stopPropagation
29
- type="text"
30
- class="flex h-11 flex-1 items-center rounded-full border bg-gray-100 px-4 text-black"
31
- placeholder="Type here"
32
- title="Type here to send a message"
33
- />
34
- <button
35
- title="Send your message"
36
- type="submit"
37
- class="rounded-full bg-black dark:bg-white dark:text-black px-4 font-semibold text-gray-200">Submit</button
38
- >
39
- </form>
 
 
 
 
 
 
 
 
 
 
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
- <img
10
- class="self-end max-h-[328px] rounded-xl bg-gray-200"
11
- src="https://huggingface.co/datasets/victor/assets/resolve/main/Frame%201.png"
12
- alt=""
13
- />
14
- <p
15
- 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"
16
- >
17
- {message.content}
18
- </p>
19
- <img
20
- class="max-h-[328px] self-start rounded-xl bg-gray-200"
21
- src="https://huggingface.co/datasets/victor/assets/resolve/main/Frame%202.png"
22
- alt=""
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 { ChatData } from './types';
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) => localforage.setItem<ChatData>('chatsStore', 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) => localforage.setItem<string>('selectedChatId', value));
43
-
44
- export { loadingState, chatsStore, selectedChatId };
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
- // export const selectedChatId = writable<string>(
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