SettingsModal.svelte 59.9 KB
Newer Older
Timothy J. Baek's avatar
Timothy J. Baek committed
1
<script lang="ts">
Timothy J. Baek's avatar
Timothy J. Baek committed
2
	import Modal from '../common/Modal.svelte';
3

Timothy J. Baek's avatar
Timothy J. Baek committed
4
5
6
7
8
9
	import {
		WEB_UI_VERSION,
		OLLAMA_API_BASE_URL,
		WEBUI_API_BASE_URL,
		WEBUI_BASE_URL
	} from '$lib/constants';
10
	import toast from 'svelte-french-toast';
Timothy J. Baek's avatar
Timothy J. Baek committed
11
	import { onMount } from 'svelte';
12
	import { config, models, settings, user, chats } from '$lib/stores';
13
	import { splitStream, getGravatarURL } from '$lib/utils';
14
	import Advanced from './Settings/Advanced.svelte';
Timothy J. Baek's avatar
Timothy J. Baek committed
15
	import { stringify } from 'postcss';
Timothy J. Baek's avatar
Timothy J. Baek committed
16
	import { getOllamaVersion } from '$lib/apis/ollama';
17
	import { createNewChat, getChatList } from '$lib/apis/chats';
18

Timothy J. Baek's avatar
Timothy J. Baek committed
19
	export let show = false;
20
21
22
23

	const saveSettings = async (updated) => {
		console.log(updated);
		await settings.set({ ...$settings, ...updated });
24
		await models.set(await getModels());
25
26
		localStorage.setItem('settings', JSON.stringify($settings));
	};
27

28
29
30
	let selectedTab = 'general';

	// General
Timothy J. Baek's avatar
Timothy J. Baek committed
31
	let API_BASE_URL = OLLAMA_API_BASE_URL;
Timothy J. Baek's avatar
Timothy J. Baek committed
32
	let theme = 'dark';
Timothy J. Baek's avatar
Timothy J. Baek committed
33
	let notificationEnabled = false;
34
	let system = '';
35
36

	// Advanced
37
	let requestFormat = '';
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
	let options = {
		// Advanced
		seed: 0,
		temperature: '',
		repeat_penalty: '',
		repeat_last_n: '',
		mirostat: '',
		mirostat_eta: '',
		mirostat_tau: '',
		top_k: '',
		top_p: '',
		stop: '',
		tfs_z: '',
		num_ctx: ''
	};
53

54
	// Models
55
56
	let modelTransferring = false;

57
	let modelTag = '';
58
59
60
61
	let digest = '';
	let pullProgress = null;

	let modelUploadMode = 'file';
Timothy J. Baek's avatar
Timothy J. Baek committed
62
	let modelInputFile = '';
63
	let modelFileUrl = '';
Timothy J. Baek's avatar
Timothy J. Baek committed
64
	let modelFileContent = `TEMPLATE """{{ .System }}\nUSER: {{ .Prompt }}\nASSSISTANT: """\nPARAMETER num_ctx 4096\nPARAMETER stop "</s>"\nPARAMETER stop "USER:"\nPARAMETER stop "ASSSISTANT:"`;
65
66
	let modelFileDigest = '';
	let uploadProgress = null;
Timothy J. Baek's avatar
Timothy J. Baek committed
67

68
69
	let deleteModelTag = '';

Timothy J. Baek's avatar
Timothy J. Baek committed
70
	// Addons
71
	let titleAutoGenerate = true;
72
	let speechAutoSend = false;
Timothy J. Baek's avatar
Timothy J. Baek committed
73
	let responseAutoCopy = false;
Timothy J. Baek's avatar
Timothy J. Baek committed
74

Timothy J. Baek's avatar
Timothy J. Baek committed
75
76
	let gravatarEmail = '';
	let OPENAI_API_KEY = '';
77
	let OPENAI_API_BASE_URL = '';
Timothy J. Baek's avatar
Timothy J. Baek committed
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
	// Chats

	let importFiles;
	let showDeleteHistoryConfirm = false;

	const importChats = async (_chats) => {
		for (const chat of _chats) {
			console.log(chat);
			await createNewChat(localStorage.token, chat);
		}

		await chats.set(await getChatList(localStorage.token));
	};

	const exportChats = async () => {
		console.log('TODO: export all chats');
	};

	$: if (importFiles) {
		console.log(importFiles);

		let reader = new FileReader();
		reader.onload = (event) => {
			let chats = JSON.parse(event.target.result);
			console.log(chats);
			importChats(chats);
		};

		reader.readAsText(importFiles[0]);
	}

Timothy J. Baek's avatar
Timothy J. Baek committed
110
111
112
113
114
	// Auth
	let authEnabled = false;
	let authType = 'Basic';
	let authContent = '';

Timothy J. Baek's avatar
Timothy J. Baek committed
115
116
117
	// About
	let ollamaVersion = '';

118
	const checkOllamaConnection = async () => {
119
		if (API_BASE_URL === '') {
Timothy J. Baek's avatar
Timothy J. Baek committed
120
			API_BASE_URL = OLLAMA_API_BASE_URL;
121
		}
Timothy J. Baek's avatar
Timothy J. Baek committed
122
		const _models = await getModels(API_BASE_URL, 'ollama');
123

Timothy J. Baek's avatar
Timothy J. Baek committed
124
		if (_models.length > 0) {
125
			toast.success('Server connection verified');
Timothy J. Baek's avatar
Timothy J. Baek committed
126
127
			await models.set(_models);

128
129
130
			saveSettings({
				API_BASE_URL: API_BASE_URL
			});
131
132
133
		}
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
134
135
136
137
138
139
140
141
142
143
144
145
146
	const toggleTheme = async () => {
		if (theme === 'dark') {
			theme = 'light';
		} else {
			theme = 'dark';
		}

		localStorage.theme = theme;

		document.documentElement.classList.remove(theme === 'dark' ? 'light' : 'dark');
		document.documentElement.classList.add(theme);
	};

147
	const toggleRequestFormat = async () => {
148
149
150
151
152
153
154
155
156
		if (requestFormat === '') {
			requestFormat = 'json';
		} else {
			requestFormat = '';
		}

		saveSettings({ requestFormat: requestFormat !== '' ? requestFormat : undefined });
	};

157
158
159
160
161
	const toggleSpeechAutoSend = async () => {
		speechAutoSend = !speechAutoSend;
		saveSettings({ speechAutoSend: speechAutoSend });
	};

162
163
164
165
166
	const toggleTitleAutoGenerate = async () => {
		titleAutoGenerate = !titleAutoGenerate;
		saveSettings({ titleAutoGenerate: titleAutoGenerate });
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
167
	const toggleNotification = async () => {
Timothy J. Baek's avatar
Timothy J. Baek committed
168
169
170
		const permission = await Notification.requestPermission();

		if (permission === 'granted') {
Timothy J. Baek's avatar
Timothy J. Baek committed
171
172
			notificationEnabled = !notificationEnabled;
			saveSettings({ notificationEnabled: notificationEnabled });
Timothy J. Baek's avatar
Timothy J. Baek committed
173
174
175
176
177
178
179
		} else {
			toast.error(
				'Response notifications cannot be activated as the website permissions have been denied. Please visit your browser settings to grant the necessary access.'
			);
		}
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
	const toggleResponseAutoCopy = async () => {
		const permission = await navigator.clipboard
			.readText()
			.then(() => {
				return 'granted';
			})
			.catch(() => {
				return '';
			});

		console.log(permission);

		if (permission === 'granted') {
			responseAutoCopy = !responseAutoCopy;
			saveSettings({ responseAutoCopy: responseAutoCopy });
		} else {
			toast.error(
				'Clipboard write permission denied. Please check your browser settings to grant the necessary access.'
			);
		}
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
202
203
204
205
	const toggleAuthHeader = async () => {
		authEnabled = !authEnabled;
	};

206
	const pullModelHandler = async () => {
207
		modelTransferring = true;
208
209
210
		const res = await fetch(`${API_BASE_URL}/pull`, {
			method: 'POST',
			headers: {
211
				'Content-Type': 'text/event-stream',
Timothy J. Baek's avatar
Timothy J. Baek committed
212
				...($settings.authHeader && { Authorization: $settings.authHeader }),
213
				...($user && { Authorization: `Bearer ${localStorage.token}` })
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
			},
			body: JSON.stringify({
				name: modelTag
			})
		});

		const reader = res.body
			.pipeThrough(new TextDecoderStream())
			.pipeThrough(splitStream('\n'))
			.getReader();

		while (true) {
			const { value, done } = await reader.read();
			if (done) break;

			try {
				let lines = value.split('\n');

				for (const line of lines) {
					if (line !== '') {
						console.log(line);
						let data = JSON.parse(line);
						console.log(data);

						if (data.error) {
							throw data.error;
						}
Timothy J. Baek's avatar
Timothy J. Baek committed
241
242
243
244

						if (data.detail) {
							throw data.detail;
						}
245
						if (data.status) {
Timothy J. Baek's avatar
Timothy J. Baek committed
246
							if (!data.digest) {
247
								toast.success(data.status);
Timothy J. Baek's avatar
Timothy J. Baek committed
248
249
250
251
252
253
254

								if (data.status === 'success') {
									const notification = new Notification(`Ollama`, {
										body: `Model '${modelTag}' has been successfully downloaded.`,
										icon: '/favicon.png'
									});
								}
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
							} else {
								digest = data.digest;
								if (data.completed) {
									pullProgress = Math.round((data.completed / data.total) * 1000) / 10;
								} else {
									pullProgress = 100;
								}
							}
						}
					}
				}
			} catch (error) {
				console.log(error);
				toast.error(error);
			}
		}

		modelTag = '';
273
274
		modelTransferring = false;

Timothy J. Baek's avatar
Timothy J. Baek committed
275
		models.set(await getModels());
276
277
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
	const calculateSHA256 = async (file) => {
		console.log(file);
		// Create a FileReader to read the file asynchronously
		const reader = new FileReader();

		// Define a promise to handle the file reading
		const readFile = new Promise((resolve, reject) => {
			reader.onload = () => resolve(reader.result);
			reader.onerror = reject;
		});

		// Read the file as an ArrayBuffer
		reader.readAsArrayBuffer(file);

		try {
			// Wait for the FileReader to finish reading the file
			const buffer = await readFile;

			// Convert the ArrayBuffer to a Uint8Array
			const uint8Array = new Uint8Array(buffer);

			// Calculate the SHA-256 hash using Web Crypto API
			const hashBuffer = await crypto.subtle.digest('SHA-256', uint8Array);

			// Convert the hash to a hexadecimal string
			const hashArray = Array.from(new Uint8Array(hashBuffer));
			const hashHex = hashArray.map((byte) => byte.toString(16).padStart(2, '0')).join('');

			return `sha256:${hashHex}`;
		} catch (error) {
			console.error('Error calculating SHA-256 hash:', error);
			throw error;
		}
	};

	const uploadModelHandler = async () => {
314
		modelTransferring = true;
315
		uploadProgress = 0;
Timothy J. Baek's avatar
Timothy J. Baek committed
316

Timothy J. Baek's avatar
Timothy J. Baek committed
317
		let uploaded = false;
318
319
		let fileResponse = null;
		let name = '';
Timothy J. Baek's avatar
Timothy J. Baek committed
320

321
322
323
324
		if (modelUploadMode === 'file') {
			const file = modelInputFile[0];
			const formData = new FormData();
			formData.append('file', file);
Timothy J. Baek's avatar
Timothy J. Baek committed
325

326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
			fileResponse = await fetch(`${WEBUI_API_BASE_URL}/utils/upload`, {
				method: 'POST',
				headers: {
					...($settings.authHeader && { Authorization: $settings.authHeader }),
					...($user && { Authorization: `Bearer ${localStorage.token}` })
				},
				body: formData
			}).catch((error) => {
				console.log(error);
				return null;
			});
		} else {
			fileResponse = await fetch(`${WEBUI_API_BASE_URL}/utils/download?url=${modelFileUrl}`, {
				method: 'GET',
				headers: {
					...($settings.authHeader && { Authorization: $settings.authHeader }),
					...($user && { Authorization: `Bearer ${localStorage.token}` })
				}
			}).catch((error) => {
				console.log(error);
				return null;
			});
		}

		if (fileResponse && fileResponse.ok) {
			const reader = fileResponse.body
Timothy J. Baek's avatar
Timothy J. Baek committed
352
353
354
355
356
357
358
359
360
361
362
363
364
365
				.pipeThrough(new TextDecoderStream())
				.pipeThrough(splitStream('\n'))
				.getReader();

			while (true) {
				const { value, done } = await reader.read();
				if (done) break;

				try {
					let lines = value.split('\n');

					for (const line of lines) {
						if (line !== '') {
							let data = JSON.parse(line.replace(/^data: /, ''));
366
367
368
369

							if (data.progress) {
								uploadProgress = data.progress;
							}
Timothy J. Baek's avatar
Timothy J. Baek committed
370
371
372
373
374
375

							if (data.error) {
								throw data.error;
							}

							if (data.done) {
376
377
								modelFileDigest = data.blob;
								name = data.name;
Timothy J. Baek's avatar
Timothy J. Baek committed
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
								uploaded = true;
							}
						}
					}
				} catch (error) {
					console.log(error);
				}
			}
		}

		if (uploaded) {
			const res = await fetch(`${$settings?.API_BASE_URL ?? OLLAMA_API_BASE_URL}/create`, {
				method: 'POST',
				headers: {
					'Content-Type': 'text/event-stream',
					...($settings.authHeader && { Authorization: $settings.authHeader }),
					...($user && { Authorization: `Bearer ${localStorage.token}` })
				},
				body: JSON.stringify({
397
398
					name: `${name}:latest`,
					modelfile: `FROM @${modelFileDigest}\n${modelFileContent}`
Timothy J. Baek's avatar
Timothy J. Baek committed
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
				})
			}).catch((err) => {
				console.log(err);
				return null;
			});

			if (res && res.ok) {
				const reader = res.body
					.pipeThrough(new TextDecoderStream())
					.pipeThrough(splitStream('\n'))
					.getReader();

				while (true) {
					const { value, done } = await reader.read();
					if (done) break;

					try {
						let lines = value.split('\n');

						for (const line of lines) {
							if (line !== '') {
								console.log(line);
								let data = JSON.parse(line);
								console.log(data);

								if (data.error) {
									throw data.error;
								}
								if (data.detail) {
									throw data.detail;
								}

								if (data.status) {
									if (
										!data.digest &&
										!data.status.includes('writing') &&
										!data.status.includes('sha256')
									) {
										toast.success(data.status);
									} else {
										if (data.digest) {
											digest = data.digest;

											if (data.completed) {
												pullProgress = Math.round((data.completed / data.total) * 1000) / 10;
											} else {
												pullProgress = 100;
											}
										}
									}
								}
							}
						}
					} catch (error) {
						console.log(error);
						toast.error(error);
					}
				}
			}
		}

460
461
462
		modelFileUrl = '';
		modelInputFile = '';
		modelTransferring = false;
463
464
		uploadProgress = null;

Timothy J. Baek's avatar
Timothy J. Baek committed
465
466
467
		models.set(await getModels());
	};

468
469
470
471
	const deleteModelHandler = async () => {
		const res = await fetch(`${API_BASE_URL}/delete`, {
			method: 'DELETE',
			headers: {
472
				'Content-Type': 'text/event-stream',
Timothy J. Baek's avatar
Timothy J. Baek committed
473
				...($settings.authHeader && { Authorization: $settings.authHeader }),
474
				...($user && { Authorization: `Bearer ${localStorage.token}` })
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
			},
			body: JSON.stringify({
				name: deleteModelTag
			})
		});

		const reader = res.body
			.pipeThrough(new TextDecoderStream())
			.pipeThrough(splitStream('\n'))
			.getReader();

		while (true) {
			const { value, done } = await reader.read();
			if (done) break;

			try {
				let lines = value.split('\n');

				for (const line of lines) {
					if (line !== '' && line !== 'null') {
						console.log(line);
						let data = JSON.parse(line);
						console.log(data);

						if (data.error) {
							throw data.error;
						}
Timothy J. Baek's avatar
Timothy J. Baek committed
502
503
504
505
						if (data.detail) {
							throw data.detail;
						}

506
507
508
509
510
511
512
513
514
515
516
517
518
						if (data.status) {
						}
					} else {
						toast.success(`Deleted ${deleteModelTag}`);
					}
				}
			} catch (error) {
				console.log(error);
				toast.error(error);
			}
		}

		deleteModelTag = '';
Timothy J. Baek's avatar
Timothy J. Baek committed
519
		models.set(await getModels());
520
	};
Timothy J. Baek's avatar
Timothy J. Baek committed
521

Timothy J. Baek's avatar
Timothy J. Baek committed
522
	const getModels = async (url = '', type = 'all') => {
523
		let models = [];
Timothy J. Baek's avatar
Timothy J. Baek committed
524
		const res = await fetch(`${url ? url : $settings?.API_BASE_URL ?? OLLAMA_API_BASE_URL}/tags`, {
525
526
527
528
			method: 'GET',
			headers: {
				Accept: 'application/json',
				'Content-Type': 'application/json',
Timothy J. Baek's avatar
Timothy J. Baek committed
529
				...($settings.authHeader && { Authorization: $settings.authHeader }),
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
				...($user && { Authorization: `Bearer ${localStorage.token}` })
			}
		})
			.then(async (res) => {
				if (!res.ok) throw await res.json();
				return res.json();
			})
			.catch((error) => {
				console.log(error);
				if ('detail' in error) {
					toast.error(error.detail);
				} else {
					toast.error('Server connection failed');
				}
				return null;
			});
		console.log(res);
Timothy J. Baek's avatar
Timothy J. Baek committed
547
548
549
550
		models.push(...(res?.models ?? []));

		// If OpenAI API Key exists
		if (type === 'all' && $settings.OPENAI_API_KEY) {
551
552
			const API_BASE_URL = $settings.OPENAI_API_BASE_URL ?? 'https://api.openai.com/v1';

Timothy J. Baek's avatar
Timothy J. Baek committed
553
			// Validate OPENAI_API_KEY
554
			const openaiModelRes = await fetch(`${API_BASE_URL}/models`, {
Timothy J. Baek's avatar
Timothy J. Baek committed
555
556
557
558
				method: 'GET',
				headers: {
					'Content-Type': 'application/json',
					Authorization: `Bearer ${$settings.OPENAI_API_KEY}`
559
				}
Timothy J. Baek's avatar
Timothy J. Baek committed
560
561
562
563
564
565
566
567
568
569
570
			})
				.then(async (res) => {
					if (!res.ok) throw await res.json();
					return res.json();
				})
				.catch((error) => {
					console.log(error);
					toast.error(`OpenAI: ${error?.error?.message ?? 'Network Problem'}`);
					return null;
				});

571
572
573
			const openAIModels = Array.isArray(openaiModelRes)
				? openaiModelRes
				: openaiModelRes?.data ?? null;
Timothy J. Baek's avatar
Timothy J. Baek committed
574
575
576
577
578
579

			models.push(
				...(openAIModels
					? [
							{ name: 'hr' },
							...openAIModels
580
581
582
583
								.map((model) => ({ name: model.id, external: true }))
								.filter((model) =>
									API_BASE_URL.includes('openai') ? model.name.includes('gpt') : true
								)
Timothy J. Baek's avatar
Timothy J. Baek committed
584
585
586
					  ]
					: [])
			);
587
		}
Timothy J. Baek's avatar
Timothy J. Baek committed
588
589

		return models;
590
591
	};

Timothy J. Baek's avatar
Timothy J. Baek committed
592
	onMount(async () => {
Timothy J. Baek's avatar
Timothy J. Baek committed
593
		let settings = JSON.parse(localStorage.getItem('settings') ?? '{}');
594
595
596
		console.log(settings);

		theme = localStorage.theme ?? 'dark';
Timothy J. Baek's avatar
Timothy J. Baek committed
597
598
		notificationEnabled = settings.notificationEnabled ?? false;

599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
		API_BASE_URL = settings.API_BASE_URL ?? OLLAMA_API_BASE_URL;
		system = settings.system ?? '';

		requestFormat = settings.requestFormat ?? '';

		options.seed = settings.seed ?? 0;
		options.temperature = settings.temperature ?? '';
		options.repeat_penalty = settings.repeat_penalty ?? '';
		options.top_k = settings.top_k ?? '';
		options.top_p = settings.top_p ?? '';
		options.num_ctx = settings.num_ctx ?? '';
		options = { ...options, ...settings.options };

		titleAutoGenerate = settings.titleAutoGenerate ?? true;
		speechAutoSend = settings.speechAutoSend ?? false;
Timothy J. Baek's avatar
Timothy J. Baek committed
614
615
		responseAutoCopy = settings.responseAutoCopy ?? false;

616
617
		gravatarEmail = settings.gravatarEmail ?? '';
		OPENAI_API_KEY = settings.OPENAI_API_KEY ?? '';
618
		OPENAI_API_BASE_URL = settings.OPENAI_API_BASE_URL ?? 'https://api.openai.com/v1';
Timothy J. Baek's avatar
Timothy J. Baek committed
619
620
621
622
623
624

		authEnabled = settings.authHeader !== undefined ? true : false;
		if (authEnabled) {
			authType = settings.authHeader.split(' ')[0];
			authContent = settings.authHeader.split(' ')[1];
		}
Timothy J. Baek's avatar
Timothy J. Baek committed
625
626
627
628
629
630
631

		ollamaVersion = await getOllamaVersion(
			API_BASE_URL ?? OLLAMA_API_BASE_URL,
			localStorage.token
		).catch((error) => {
			return '';
		});
Timothy J. Baek's avatar
Timothy J. Baek committed
632
	});
Timothy J. Baek's avatar
Timothy J. Baek committed
633
634
635
</script>

<Modal bind:show>
Timothy J. Baek's avatar
Timothy J. Baek committed
636
	<div>
Timothy J. Baek's avatar
Timothy J. Baek committed
637
		<div class=" flex justify-between dark:text-gray-300 px-5 py-4">
Timothy J. Baek's avatar
Timothy J. Baek committed
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
			<div class=" text-lg font-medium self-center">Settings</div>
			<button
				class="self-center"
				on:click={() => {
					show = false;
				}}
			>
				<svg
					xmlns="http://www.w3.org/2000/svg"
					viewBox="0 0 20 20"
					fill="currentColor"
					class="w-5 h-5"
				>
					<path
						d="M6.28 5.22a.75.75 0 00-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 101.06 1.06L10 11.06l3.72 3.72a.75.75 0 101.06-1.06L11.06 10l3.72-3.72a.75.75 0 00-1.06-1.06L10 8.94 6.28 5.22z"
					/>
				</svg>
			</button>
		</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
657
		<hr class=" dark:border-gray-800" />
Timothy J. Baek's avatar
Timothy J. Baek committed
658

Timothy J. Baek's avatar
Timothy J. Baek committed
659
660
		<div class="flex flex-col md:flex-row w-full p-4 md:space-x-4">
			<div
Timothy J. Baek's avatar
Timothy J. Baek committed
661
				class="tabs flex flex-row overflow-x-auto space-x-1 md:space-x-0 md:space-y-1 md:flex-col flex-1 md:flex-none md:w-40 dark:text-gray-200 text-xs text-left mb-3 md:mb-0"
Timothy J. Baek's avatar
Timothy J. Baek committed
662
663
			>
				<button
664
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
Timothy J. Baek's avatar
Timothy J. Baek committed
665
					'general'
Timothy J. Baek's avatar
Timothy J. Baek committed
666
667
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
Timothy J. Baek's avatar
Timothy J. Baek committed
668
					on:click={() => {
669
						selectedTab = 'general';
Timothy J. Baek's avatar
Timothy J. Baek committed
670
					}}
Timothy J. Baek's avatar
Timothy J. Baek committed
671
				>
Timothy J. Baek's avatar
Timothy J. Baek committed
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 20 20"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								fill-rule="evenodd"
								d="M8.34 1.804A1 1 0 019.32 1h1.36a1 1 0 01.98.804l.295 1.473c.497.144.971.342 1.416.587l1.25-.834a1 1 0 011.262.125l.962.962a1 1 0 01.125 1.262l-.834 1.25c.245.445.443.919.587 1.416l1.473.294a1 1 0 01.804.98v1.361a1 1 0 01-.804.98l-1.473.295a6.95 6.95 0 01-.587 1.416l.834 1.25a1 1 0 01-.125 1.262l-.962.962a1 1 0 01-1.262.125l-1.25-.834a6.953 6.953 0 01-1.416.587l-.294 1.473a1 1 0 01-.98.804H9.32a1 1 0 01-.98-.804l-.295-1.473a6.957 6.957 0 01-1.416-.587l-1.25.834a1 1 0 01-1.262-.125l-.962-.962a1 1 0 01-.125-1.262l.834-1.25a6.957 6.957 0 01-.587-1.416l-1.473-.294A1 1 0 011 10.68V9.32a1 1 0 01.804-.98l1.473-.295c.144-.497.342-.971.587-1.416l-.834-1.25a1 1 0 01.125-1.262l.962-.962A1 1 0 015.38 3.03l1.25.834a6.957 6.957 0 011.416-.587l.294-1.473zM13 10a3 3 0 11-6 0 3 3 0 016 0z"
								clip-rule="evenodd"
							/>
						</svg>
					</div>
					<div class=" self-center">General</div>
				</button>

Timothy J. Baek's avatar
Timothy J. Baek committed
689
				<button
690
691
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
					'advanced'
Timothy J. Baek's avatar
Timothy J. Baek committed
692
693
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
					on:click={() => {
						selectedTab = 'advanced';
					}}
				>
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 20 20"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								d="M17 2.75a.75.75 0 00-1.5 0v5.5a.75.75 0 001.5 0v-5.5zM17 15.75a.75.75 0 00-1.5 0v1.5a.75.75 0 001.5 0v-1.5zM3.75 15a.75.75 0 01.75.75v1.5a.75.75 0 01-1.5 0v-1.5a.75.75 0 01.75-.75zM4.5 2.75a.75.75 0 00-1.5 0v5.5a.75.75 0 001.5 0v-5.5zM10 11a.75.75 0 01.75.75v5.5a.75.75 0 01-1.5 0v-5.5A.75.75 0 0110 11zM10.75 2.75a.75.75 0 00-1.5 0v1.5a.75.75 0 001.5 0v-1.5zM10 6a2 2 0 100 4 2 2 0 000-4zM3.75 10a2 2 0 100 4 2 2 0 000-4zM16.25 10a2 2 0 100 4 2 2 0 000-4z"
							/>
						</svg>
					</div>
					<div class=" self-center">Advanced</div>
				</button>

				<button
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
Timothy J. Baek's avatar
Timothy J. Baek committed
715
					'models'
Timothy J. Baek's avatar
Timothy J. Baek committed
716
717
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
Timothy J. Baek's avatar
Timothy J. Baek committed
718
					on:click={() => {
719
						selectedTab = 'models';
Timothy J. Baek's avatar
Timothy J. Baek committed
720
721
					}}
				>
Timothy J. Baek's avatar
Timothy J. Baek committed
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 20 20"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								fill-rule="evenodd"
								d="M10 1c3.866 0 7 1.79 7 4s-3.134 4-7 4-7-1.79-7-4 3.134-4 7-4zm5.694 8.13c.464-.264.91-.583 1.306-.952V10c0 2.21-3.134 4-7 4s-7-1.79-7-4V8.178c.396.37.842.688 1.306.953C5.838 10.006 7.854 10.5 10 10.5s4.162-.494 5.694-1.37zM3 13.179V15c0 2.21 3.134 4 7 4s7-1.79 7-4v-1.822c-.396.37-.842.688-1.306.953-1.532.875-3.548 1.369-5.694 1.369s-4.162-.494-5.694-1.37A7.009 7.009 0 013 13.179z"
								clip-rule="evenodd"
							/>
						</svg>
					</div>
					<div class=" self-center">Models</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
737
				</button>
738

739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
				<button
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
					'external'
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
					on:click={() => {
						selectedTab = 'external';
					}}
				>
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 16 16"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								d="M1 9.5A3.5 3.5 0 0 0 4.5 13H12a3 3 0 0 0 .917-5.857 2.503 2.503 0 0 0-3.198-3.019 3.5 3.5 0 0 0-6.628 2.171A3.5 3.5 0 0 0 1 9.5Z"
							/>
						</svg>
					</div>
					<div class=" self-center">External</div>
				</button>

763
				<button
764
765
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
					'addons'
Timothy J. Baek's avatar
Timothy J. Baek committed
766
767
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
768
					on:click={() => {
769
						selectedTab = 'addons';
770
771
772
773
774
775
776
777
778
779
780
781
782
783
					}}
				>
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 20 20"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								d="M12 4.467c0-.405.262-.75.559-1.027.276-.257.441-.584.441-.94 0-.828-.895-1.5-2-1.5s-2 .672-2 1.5c0 .362.171.694.456.953.29.265.544.6.544.994a.968.968 0 01-1.024.974 39.655 39.655 0 01-3.014-.306.75.75 0 00-.847.847c.14.993.242 1.999.306 3.014A.968.968 0 014.447 10c-.393 0-.729-.253-.994-.544C3.194 9.17 2.862 9 2.5 9 1.672 9 1 9.895 1 11s.672 2 1.5 2c.356 0 .683-.165.94-.441.276-.297.622-.559 1.027-.559a.997.997 0 011.004 1.03 39.747 39.747 0 01-.319 3.734.75.75 0 00.64.842c1.05.146 2.111.252 3.184.318A.97.97 0 0010 16.948c0-.394-.254-.73-.545-.995C9.171 15.693 9 15.362 9 15c0-.828.895-1.5 2-1.5s2 .672 2 1.5c0 .356-.165.683-.441.94-.297.276-.559.622-.559 1.027a.998.998 0 001.03 1.005c1.337-.05 2.659-.162 3.961-.337a.75.75 0 00.644-.644c.175-1.302.288-2.624.337-3.961A.998.998 0 0016.967 12c-.405 0-.75.262-1.027.559-.257.276-.584.441-.94.441-.828 0-1.5-.895-1.5-2s.672-2 1.5-2c.362 0 .694.17.953.455.265.291.601.545.995.545a.97.97 0 00.976-1.024 41.159 41.159 0 00-.318-3.184.75.75 0 00-.842-.64c-1.228.164-2.473.271-3.734.319A.997.997 0 0112 4.467z"
							/>
						</svg>
					</div>
784
					<div class=" self-center">Add-ons</div>
785
				</button>
Timothy J. Baek's avatar
Timothy J. Baek committed
786

787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
				<button
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
					'chats'
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
					on:click={() => {
						selectedTab = 'chats';
					}}
				>
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 16 16"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								fill-rule="evenodd"
								d="M8 2C4.262 2 1 4.57 1 8c0 1.86.98 3.486 2.455 4.566a3.472 3.472 0 0 1-.469 1.26.75.75 0 0 0 .713 1.14 6.961 6.961 0 0 0 3.06-1.06c.403.062.818.094 1.241.094 3.738 0 7-2.57 7-6s-3.262-6-7-6ZM5 9a1 1 0 1 0 0-2 1 1 0 0 0 0 2Zm7-1a1 1 0 1 1-2 0 1 1 0 0 1 2 0ZM8 9a1 1 0 1 0 0-2 1 1 0 0 0 0 2Z"
								clip-rule="evenodd"
							/>
						</svg>
					</div>
					<div class=" self-center">Chats</div>
				</button>

Timothy J. Baek's avatar
Timothy J. Baek committed
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
				{#if !$config || ($config && !$config.auth)}
					<button
						class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
						'auth'
							? 'bg-gray-200 dark:bg-gray-700'
							: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
						on:click={() => {
							selectedTab = 'auth';
						}}
					>
						<div class=" self-center mr-2">
							<svg
								xmlns="http://www.w3.org/2000/svg"
								viewBox="0 0 24 24"
								fill="currentColor"
								class="w-4 h-4"
							>
								<path
									fill-rule="evenodd"
									d="M12.516 2.17a.75.75 0 00-1.032 0 11.209 11.209 0 01-7.877 3.08.75.75 0 00-.722.515A12.74 12.74 0 002.25 9.75c0 5.942 4.064 10.933 9.563 12.348a.749.749 0 00.374 0c5.499-1.415 9.563-6.406 9.563-12.348 0-1.39-.223-2.73-.635-3.985a.75.75 0 00-.722-.516l-.143.001c-2.996 0-5.717-1.17-7.734-3.08zm3.094 8.016a.75.75 0 10-1.22-.872l-3.236 4.53L9.53 12.22a.75.75 0 00-1.06 1.06l2.25 2.25a.75.75 0 001.14-.094l3.75-5.25z"
									clip-rule="evenodd"
								/>
							</svg>
						</div>
						<div class=" self-center">Authentication</div>
					</button>
				{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
840

Timothy J. Baek's avatar
Timothy J. Baek committed
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
				<button
					class="px-2.5 py-2.5 min-w-fit rounded-lg flex-1 md:flex-none flex text-right transition {selectedTab ===
					'about'
						? 'bg-gray-200 dark:bg-gray-700'
						: ' hover:bg-gray-300 dark:hover:bg-gray-800'}"
					on:click={() => {
						selectedTab = 'about';
					}}
				>
					<div class=" self-center mr-2">
						<svg
							xmlns="http://www.w3.org/2000/svg"
							viewBox="0 0 20 20"
							fill="currentColor"
							class="w-4 h-4"
						>
							<path
								fill-rule="evenodd"
								d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7-4a1 1 0 11-2 0 1 1 0 012 0zM9 9a.75.75 0 000 1.5h.253a.25.25 0 01.244.304l-.459 2.066A1.75 1.75 0 0010.747 15H11a.75.75 0 000-1.5h-.253a.25.25 0 01-.244-.304l.459-2.066A1.75 1.75 0 009.253 9H9z"
								clip-rule="evenodd"
							/>
						</svg>
					</div>
					<div class=" self-center">About</div>
				</button>
Timothy J. Baek's avatar
Timothy J. Baek committed
866
			</div>
867
			<div class="flex-1 md:min-h-[340px]">
868
				{#if selectedTab === 'general'}
Timothy J. Baek's avatar
Timothy J. Baek committed
869
					<div class="flex flex-col space-y-3">
Timothy J. Baek's avatar
Timothy J. Baek committed
870
						<div>
Timothy J. Baek's avatar
Timothy J. Baek committed
871
872
873
874
							<div class=" mb-1 text-sm font-medium">WebUI Settings</div>

							<div class=" py-0.5 flex w-full justify-between">
								<div class=" self-center text-xs font-medium">Theme</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911

								<button
									class="p-1 px-3 text-xs flex rounded transition"
									on:click={() => {
										toggleTheme();
									}}
								>
									{#if theme === 'dark'}
										<svg
											xmlns="http://www.w3.org/2000/svg"
											viewBox="0 0 20 20"
											fill="currentColor"
											class="w-4 h-4"
										>
											<path
												fill-rule="evenodd"
												d="M7.455 2.004a.75.75 0 01.26.77 7 7 0 009.958 7.967.75.75 0 011.067.853A8.5 8.5 0 116.647 1.921a.75.75 0 01.808.083z"
												clip-rule="evenodd"
											/>
										</svg>

										<span class="ml-2 self-center"> Dark </span>
									{:else}
										<svg
											xmlns="http://www.w3.org/2000/svg"
											viewBox="0 0 20 20"
											fill="currentColor"
											class="w-4 h-4 self-center"
										>
											<path
												d="M10 2a.75.75 0 01.75.75v1.5a.75.75 0 01-1.5 0v-1.5A.75.75 0 0110 2zM10 15a.75.75 0 01.75.75v1.5a.75.75 0 01-1.5 0v-1.5A.75.75 0 0110 15zM10 7a3 3 0 100 6 3 3 0 000-6zM15.657 5.404a.75.75 0 10-1.06-1.06l-1.061 1.06a.75.75 0 001.06 1.06l1.06-1.06zM6.464 14.596a.75.75 0 10-1.06-1.06l-1.06 1.06a.75.75 0 001.06 1.06l1.06-1.06zM18 10a.75.75 0 01-.75.75h-1.5a.75.75 0 010-1.5h1.5A.75.75 0 0118 10zM5 10a.75.75 0 01-.75.75h-1.5a.75.75 0 010-1.5h1.5A.75.75 0 015 10zM14.596 15.657a.75.75 0 001.06-1.06l-1.06-1.061a.75.75 0 10-1.06 1.06l1.06 1.06zM5.404 6.464a.75.75 0 001.06-1.06l-1.06-1.06a.75.75 0 10-1.061 1.06l1.06 1.06z"
											/>
										</svg>
										<span class="ml-2 self-center"> Light </span>
									{/if}
								</button>
							</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931

							<div>
								<div class=" py-0.5 flex w-full justify-between">
									<div class=" self-center text-xs font-medium">Notification</div>

									<button
										class="p-1 px-3 text-xs flex rounded transition"
										on:click={() => {
											toggleNotification();
										}}
										type="button"
									>
										{#if notificationEnabled === true}
											<span class="ml-2 self-center">On</span>
										{:else}
											<span class="ml-2 self-center">Off</span>
										{/if}
									</button>
								</div>
							</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
932
933
934
						</div>

						<hr class=" dark:border-gray-700" />
935
936
937
938
939
						<div>
							<div class=" mb-2.5 text-sm font-medium">Ollama Server URL</div>
							<div class="flex w-full">
								<div class="flex-1 mr-2">
									<input
Timothy J. Baek's avatar
Timothy J. Baek committed
940
										class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
941
942
943
944
945
										placeholder="Enter URL (e.g. http://localhost:11434/api)"
										bind:value={API_BASE_URL}
									/>
								</div>
								<button
Timothy J. Baek's avatar
Timothy J. Baek committed
946
									class="px-3 bg-gray-200 hover:bg-gray-300 dark:bg-gray-600 dark:hover:bg-gray-700 rounded transition"
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
									on:click={() => {
										checkOllamaConnection();
									}}
								>
									<svg
										xmlns="http://www.w3.org/2000/svg"
										viewBox="0 0 20 20"
										fill="currentColor"
										class="w-4 h-4"
									>
										<path
											fill-rule="evenodd"
											d="M15.312 11.424a5.5 5.5 0 01-9.201 2.466l-.312-.311h2.433a.75.75 0 000-1.5H3.989a.75.75 0 00-.75.75v4.242a.75.75 0 001.5 0v-2.43l.31.31a7 7 0 0011.712-3.138.75.75 0 00-1.449-.39zm1.23-3.723a.75.75 0 00.219-.53V2.929a.75.75 0 00-1.5 0V5.36l-.31-.31A7 7 0 003.239 8.188a.75.75 0 101.448.389A5.5 5.5 0 0113.89 6.11l.311.31h-2.432a.75.75 0 000 1.5h4.243a.75.75 0 00.53-.219z"
											clip-rule="evenodd"
										/>
									</svg>
								</button>
							</div>

Timothy J. Baek's avatar
Timothy J. Baek committed
966
							<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
967
								Trouble accessing Ollama? <a
Timothy J. Baek's avatar
Timothy J. Baek committed
968
									class=" text-gray-500 dark:text-gray-300 font-medium"
969
970
971
972
973
974
975
976
									href="https://github.com/ollama-webui/ollama-webui#troubleshooting"
									target="_blank"
								>
									Click here for help.
								</a>
							</div>
						</div>

Timothy J. Baek's avatar
Timothy J. Baek committed
977
						<hr class=" dark:border-gray-700" />
978

Timothy J. Baek's avatar
Timothy J. Baek committed
979
						<div>
980
981
982
							<div class=" mb-2.5 text-sm font-medium">System Prompt</div>
							<textarea
								bind:value={system}
Timothy J. Baek's avatar
Timothy J. Baek committed
983
								class="w-full rounded p-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none resize-none"
984
								rows="4"
Timothy J. Baek's avatar
Timothy J. Baek committed
985
986
							/>
						</div>
987

Timothy J. Baek's avatar
Timothy J. Baek committed
988
989
						<div class="flex justify-end pt-3 text-sm font-medium">
							<button
Timothy J. Baek's avatar
Timothy J. Baek committed
990
								class=" px-4 py-2 bg-emerald-600 hover:bg-emerald-700 text-gray-100 transition rounded"
Timothy J. Baek's avatar
Timothy J. Baek committed
991
								on:click={() => {
992
									saveSettings({
Timothy J. Baek's avatar
Timothy J. Baek committed
993
										API_BASE_URL: API_BASE_URL === '' ? OLLAMA_API_BASE_URL : API_BASE_URL,
994
995
										system: system !== '' ? system : undefined
									});
Timothy J. Baek's avatar
Timothy J. Baek committed
996
997
998
999
1000
1001
1002
									show = false;
								}}
							>
								Save
							</button>
						</div>
					</div>
1003
				{:else if selectedTab === 'advanced'}
1004
1005
1006
					<div class="flex flex-col h-full justify-between text-sm">
						<div class=" space-y-3 pr-1.5 overflow-y-scroll max-h-72">
							<div class=" text-sm font-medium">Parameters</div>
1007

1008
							<Advanced bind:options />
1009
1010
							<hr class=" dark:border-gray-700" />

1011
							<div>
1012
1013
1014
1015
1016
1017
								<div class=" py-1 flex w-full justify-between">
									<div class=" self-center text-sm font-medium">Request Mode</div>

									<button
										class="p-1 px-3 text-xs flex rounded transition"
										on:click={() => {
1018
											toggleRequestFormat();
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
										}}
									>
										{#if requestFormat === ''}
											<span class="ml-2 self-center"> Default </span>
										{:else if requestFormat === 'json'}
											<!-- <svg
												xmlns="http://www.w3.org/2000/svg"
												viewBox="0 0 20 20"
												fill="currentColor"
												class="w-4 h-4 self-center"
											>
												<path
													d="M10 2a.75.75 0 01.75.75v1.5a.75.75 0 01-1.5 0v-1.5A.75.75 0 0110 2zM10 15a.75.75 0 01.75.75v1.5a.75.75 0 01-1.5 0v-1.5A.75.75 0 0110 15zM10 7a3 3 0 100 6 3 3 0 000-6zM15.657 5.404a.75.75 0 10-1.06-1.06l-1.061 1.06a.75.75 0 001.06 1.06l1.06-1.06zM6.464 14.596a.75.75 0 10-1.06-1.06l-1.06 1.06a.75.75 0 001.06 1.06l1.06-1.06zM18 10a.75.75 0 01-.75.75h-1.5a.75.75 0 010-1.5h1.5A.75.75 0 0118 10zM5 10a.75.75 0 01-.75.75h-1.5a.75.75 0 010-1.5h1.5A.75.75 0 015 10zM14.596 15.657a.75.75 0 001.06-1.06l-1.06-1.061a.75.75 0 10-1.06 1.06l1.06 1.06zM5.404 6.464a.75.75 0 001.06-1.06l-1.06-1.06a.75.75 0 10-1.061 1.06l1.06 1.06z"
												/>
											</svg> -->
											<span class="ml-2 self-center"> JSON </span>
										{/if}
									</button>
								</div>
							</div>
1039
						</div>
1040

1041
1042
						<div class="flex justify-end pt-3 text-sm font-medium">
							<button
Timothy J. Baek's avatar
Timothy J. Baek committed
1043
								class=" px-4 py-2 bg-emerald-600 hover:bg-emerald-700 text-gray-100 transition rounded"
1044
1045
								on:click={() => {
									saveSettings({
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
										options: {
											seed: (options.seed !== 0 ? options.seed : undefined) ?? undefined,
											stop: options.stop !== '' ? options.stop : undefined,
											temperature: options.temperature !== '' ? options.temperature : undefined,
											repeat_penalty:
												options.repeat_penalty !== '' ? options.repeat_penalty : undefined,
											repeat_last_n:
												options.repeat_last_n !== '' ? options.repeat_last_n : undefined,
											mirostat: options.mirostat !== '' ? options.mirostat : undefined,
											mirostat_eta: options.mirostat_eta !== '' ? options.mirostat_eta : undefined,
											mirostat_tau: options.mirostat_tau !== '' ? options.mirostat_tau : undefined,
											top_k: options.top_k !== '' ? options.top_k : undefined,
											top_p: options.top_p !== '' ? options.top_p : undefined,
											tfs_z: options.tfs_z !== '' ? options.tfs_z : undefined,
											num_ctx: options.num_ctx !== '' ? options.num_ctx : undefined
										}
1062
1063
1064
1065
1066
1067
1068
1069
									});
									show = false;
								}}
							>
								Save
							</button>
						</div>
					</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1070
				{:else if selectedTab === 'models'}
Timothy J. Baek's avatar
Timothy J. Baek committed
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
					<div class="flex flex-col h-full justify-between text-sm">
						<div class=" space-y-3 pr-1.5 overflow-y-scroll h-80">
							<div>
								<div class=" mb-2.5 text-sm font-medium">Pull a model from Ollama.ai</div>
								<div class="flex w-full">
									<div class="flex-1 mr-2">
										<input
											class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
											placeholder="Enter model tag (e.g. mistral:7b)"
											bind:value={modelTag}
										/>
									</div>
									<button
1084
										class="px-3 text-gray-100 bg-emerald-600 hover:bg-emerald-700 disabled:bg-gray-700 disabled:cursor-not-allowed rounded transition"
Timothy J. Baek's avatar
Timothy J. Baek committed
1085
1086
1087
										on:click={() => {
											pullModelHandler();
										}}
1088
										disabled={modelTransferring}
Timothy J. Baek's avatar
Timothy J. Baek committed
1089
									>
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
										{#if modelTransferring}
											<div class="self-center">
												<svg
													class=" w-4 h-4"
													viewBox="0 0 24 24"
													fill="currentColor"
													xmlns="http://www.w3.org/2000/svg"
													><style>
														.spinner_ajPY {
															transform-origin: center;
															animation: spinner_AtaB 0.75s infinite linear;
														}
														@keyframes spinner_AtaB {
															100% {
																transform: rotate(360deg);
															}
														}
													</style><path
														d="M12,1A11,11,0,1,0,23,12,11,11,0,0,0,12,1Zm0,19a8,8,0,1,1,8-8A8,8,0,0,1,12,20Z"
														opacity=".25"
													/><path
														d="M10.14,1.16a11,11,0,0,0-9,8.92A1.59,1.59,0,0,0,2.46,12,1.52,1.52,0,0,0,4.11,10.7a8,8,0,0,1,6.66-6.61A1.42,1.42,0,0,0,12,2.69h0A1.57,1.57,0,0,0,10.14,1.16Z"
														class="spinner_ajPY"
													/></svg
												>
											</div>
										{:else}
											<svg
												xmlns="http://www.w3.org/2000/svg"
												viewBox="0 0 16 16"
												fill="currentColor"
												class="w-4 h-4"
											>
												<path
													d="M8.75 2.75a.75.75 0 0 0-1.5 0v5.69L5.03 6.22a.75.75 0 0 0-1.06 1.06l3.5 3.5a.75.75 0 0 0 1.06 0l3.5-3.5a.75.75 0 0 0-1.06-1.06L8.75 8.44V2.75Z"
												/>
												<path
													d="M3.5 9.75a.75.75 0 0 0-1.5 0v1.5A2.75 2.75 0 0 0 4.75 14h6.5A2.75 2.75 0 0 0 14 11.25v-1.5a.75.75 0 0 0-1.5 0v1.5c0 .69-.56 1.25-1.25 1.25h-6.5c-.69 0-1.25-.56-1.25-1.25v-1.5Z"
												/>
											</svg>
										{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
1131
									</button>
Timothy J. Baek's avatar
Timothy J. Baek committed
1132
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1133
1134
1135
1136
1137
1138

								<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
									To access the available model names for downloading, <a
										class=" text-gray-500 dark:text-gray-300 font-medium"
										href="https://ollama.ai/library"
										target="_blank">click here.</a
Timothy J. Baek's avatar
Timothy J. Baek committed
1139
									>
Timothy J. Baek's avatar
Timothy J. Baek committed
1140
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1141

Timothy J. Baek's avatar
Timothy J. Baek committed
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
								{#if pullProgress !== null}
									<div class="mt-2">
										<div class=" mb-2 text-xs">Pull Progress</div>
										<div class="w-full rounded-full dark:bg-gray-800">
											<div
												class="dark:bg-gray-600 text-xs font-medium text-blue-100 text-center p-0.5 leading-none rounded-full"
												style="width: {Math.max(15, pullProgress ?? 0)}%"
											>
												{pullProgress ?? 0}%
											</div>
										</div>
										<div class="mt-1 text-xs dark:text-gray-500" style="font-size: 0.5rem;">
											{digest}
										</div>
									</div>
								{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
1158
							</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1159
1160
							<hr class=" dark:border-gray-700" />

1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
							<form
								on:submit|preventDefault={() => {
									uploadModelHandler();
								}}
							>
								<div class=" mb-2 flex w-full justify-between">
									<div class="  text-sm font-medium">Upload a GGUF model</div>

									<button
										class="p-1 px-3 text-xs flex rounded transition"
										on:click={() => {
											if (modelUploadMode === 'file') {
												modelUploadMode = 'url';
											} else {
												modelUploadMode = 'file';
											}
										}}
										type="button"
									>
										{#if modelUploadMode === 'file'}
											<span class="ml-2 self-center">File Mode</span>
										{:else}
											<span class="ml-2 self-center">URL Mode</span>
										{/if}
									</button>
								</div>

Timothy J. Baek's avatar
Timothy J. Baek committed
1188
								<div class="flex w-full mb-1.5">
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
									<div class="flex flex-col w-full">
										{#if modelUploadMode === 'file'}
											<div
												class="flex-1 {modelInputFile && modelInputFile.length > 0 ? 'mr-2' : ''}"
											>
												<input
													id="model-upload-input"
													type="file"
													bind:files={modelInputFile}
													on:change={() => {
														console.log(modelInputFile);
													}}
													accept=".gguf"
													required
													hidden
												/>
Timothy J. Baek's avatar
Timothy J. Baek committed
1205

1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
												<button
													type="button"
													class="w-full rounded text-left py-2 px-4 dark:text-gray-300 dark:bg-gray-800"
													on:click={() => {
														document.getElementById('model-upload-input').click();
													}}
												>
													{#if modelInputFile && modelInputFile.length > 0}
														{modelInputFile[0].name}
													{:else}
														Click here to select
													{/if}
												</button>
											</div>
										{:else}
											<div class="flex-1 {modelFileUrl !== '' ? 'mr-2' : ''}">
												<input
													class="w-full rounded text-left py-2 px-4 dark:text-gray-300 dark:bg-gray-800 outline-none {modelFileUrl !==
													''
														? 'mr-2'
														: ''}"
													type="url"
													required
													bind:value={modelFileUrl}
													placeholder="Type HuggingFace Resolve (Download) URL"
												/>
											</div>
										{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
1234
									</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1235

1236
									{#if (modelUploadMode === 'file' && modelInputFile && modelInputFile.length > 0) || (modelUploadMode === 'url' && modelFileUrl !== '')}
Timothy J. Baek's avatar
Timothy J. Baek committed
1237
										<button
1238
1239
1240
											class="px-3 text-gray-100 bg-emerald-600 hover:bg-emerald-700 disabled:bg-gray-700 disabled:cursor-not-allowed rounded transition"
											type="submit"
											disabled={modelTransferring}
Timothy J. Baek's avatar
Timothy J. Baek committed
1241
										>
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
											{#if modelTransferring}
												<div class="self-center">
													<svg
														class=" w-4 h-4"
														viewBox="0 0 24 24"
														fill="currentColor"
														xmlns="http://www.w3.org/2000/svg"
														><style>
															.spinner_ajPY {
																transform-origin: center;
																animation: spinner_AtaB 0.75s infinite linear;
															}
															@keyframes spinner_AtaB {
																100% {
																	transform: rotate(360deg);
																}
															}
														</style><path
															d="M12,1A11,11,0,1,0,23,12,11,11,0,0,0,12,1Zm0,19a8,8,0,1,1,8-8A8,8,0,0,1,12,20Z"
															opacity=".25"
														/><path
															d="M10.14,1.16a11,11,0,0,0-9,8.92A1.59,1.59,0,0,0,2.46,12,1.52,1.52,0,0,0,4.11,10.7a8,8,0,0,1,6.66-6.61A1.42,1.42,0,0,0,12,2.69h0A1.57,1.57,0,0,0,10.14,1.16Z"
															class="spinner_ajPY"
														/></svg
													>
												</div>
											{:else}
												<svg
													xmlns="http://www.w3.org/2000/svg"
													viewBox="0 0 16 16"
													fill="currentColor"
													class="w-4 h-4"
												>
													<path
														d="M7.25 10.25a.75.75 0 0 0 1.5 0V4.56l2.22 2.22a.75.75 0 1 0 1.06-1.06l-3.5-3.5a.75.75 0 0 0-1.06 0l-3.5 3.5a.75.75 0 0 0 1.06 1.06l2.22-2.22v5.69Z"
													/>
													<path
														d="M3.5 9.75a.75.75 0 0 0-1.5 0v1.5A2.75 2.75 0 0 0 4.75 14h6.5A2.75 2.75 0 0 0 14 11.25v-1.5a.75.75 0 0 0-1.5 0v1.5c0 .69-.56 1.25-1.25 1.25h-6.5c-.69 0-1.25-.56-1.25-1.25v-1.5Z"
													/>
												</svg>
											{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
1283
1284
										</button>
									{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
1285
1286
								</div>

1287
								{#if (modelUploadMode === 'file' && modelInputFile && modelInputFile.length > 0) || (modelUploadMode === 'url' && modelFileUrl !== '')}
Timothy J. Baek's avatar
Timothy J. Baek committed
1288
1289
1290
1291
1292
1293
									<div>
										<div>
											<div class=" my-2.5 text-sm font-medium">Modelfile Content</div>
											<textarea
												bind:value={modelFileContent}
												class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none resize-none"
Timothy J. Baek's avatar
Timothy J. Baek committed
1294
												rows="6"
Timothy J. Baek's avatar
Timothy J. Baek committed
1295
1296
1297
1298
1299
1300
1301
1302
1303
											/>
										</div>
									</div>
								{/if}
								<div class=" mt-1 text-xs text-gray-400 dark:text-gray-500">
									To access the GGUF models available for downloading, <a
										class=" text-gray-500 dark:text-gray-300 font-medium"
										href="https://huggingface.co/models?search=gguf"
										target="_blank">click here.</a
1304
									>
Timothy J. Baek's avatar
Timothy J. Baek committed
1305
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1306

1307
								{#if uploadProgress !== null}
Timothy J. Baek's avatar
Timothy J. Baek committed
1308
									<div class="mt-2">
1309
										<div class=" mb-2 text-xs">Upload Progress</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1310
1311
1312
										<div class="w-full rounded-full dark:bg-gray-800">
											<div
												class="dark:bg-gray-600 text-xs font-medium text-blue-100 text-center p-0.5 leading-none rounded-full"
1313
												style="width: {Math.max(15, uploadProgress ?? 0)}%"
Timothy J. Baek's avatar
Timothy J. Baek committed
1314
											>
1315
												{uploadProgress ?? 0}%
Timothy J. Baek's avatar
Timothy J. Baek committed
1316
1317
1318
											</div>
										</div>
										<div class="mt-1 text-xs dark:text-gray-500" style="font-size: 0.5rem;">
1319
											{modelFileDigest}
Timothy J. Baek's avatar
Timothy J. Baek committed
1320
1321
1322
										</div>
									</div>
								{/if}
1323
							</form>
Timothy J. Baek's avatar
Timothy J. Baek committed
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
							<hr class=" dark:border-gray-700" />

							<div>
								<div class=" mb-2.5 text-sm font-medium">Delete a model</div>
								<div class="flex w-full">
									<div class="flex-1 mr-2">
										<select
											class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
											bind:value={deleteModelTag}
											placeholder="Select a model"
										>
											{#if !deleteModelTag}
												<option value="" disabled selected>Select a model</option>
											{/if}
											{#each $models.filter((m) => m.size != null) as model}
												<option value={model.name} class="bg-gray-100 dark:bg-gray-700"
													>{model.name +
														' (' +
														(model.size / 1024 ** 3).toFixed(1) +
														' GB)'}</option
												>
											{/each}
										</select>
									</div>
									<button
										class="px-3 bg-red-700 hover:bg-red-800 text-gray-100 rounded transition"
										on:click={() => {
											deleteModelHandler();
										}}
Timothy J. Baek's avatar
Timothy J. Baek committed
1353
									>
Timothy J. Baek's avatar
Timothy J. Baek committed
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
										<svg
											xmlns="http://www.w3.org/2000/svg"
											viewBox="0 0 16 16"
											fill="currentColor"
											class="w-4 h-4"
										>
											<path
												fill-rule="evenodd"
												d="M5 3.25V4H2.75a.75.75 0 0 0 0 1.5h.3l.815 8.15A1.5 1.5 0 0 0 5.357 15h5.285a1.5 1.5 0 0 0 1.493-1.35l.815-8.15h.3a.75.75 0 0 0 0-1.5H11v-.75A2.25 2.25 0 0 0 8.75 1h-1.5A2.25 2.25 0 0 0 5 3.25Zm2.25-.75a.75.75 0 0 0-.75.75V4h3v-.75a.75.75 0 0 0-.75-.75h-1.5ZM6.05 6a.75.75 0 0 1 .787.713l.275 5.5a.75.75 0 0 1-1.498.075l-.275-5.5A.75.75 0 0 1 6.05 6Zm3.9 0a.75.75 0 0 1 .712.787l-.275 5.5a.75.75 0 0 1-1.498-.075l.275-5.5a.75.75 0 0 1 .786-.711Z"
												clip-rule="evenodd"
											/>
										</svg>
									</button>
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1368
1369
1370
							</div>
						</div>
					</div>
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
				{:else if selectedTab === 'external'}
					<form
						class="flex flex-col h-full justify-between space-y-3 text-sm"
						on:submit|preventDefault={() => {
							saveSettings({
								OPENAI_API_KEY: OPENAI_API_KEY !== '' ? OPENAI_API_KEY : undefined,
								OPENAI_API_BASE_URL: OPENAI_API_BASE_URL !== '' ? OPENAI_API_BASE_URL : undefined
							});
							show = false;
						}}
					>
						<div class=" space-y-3">
							<div>
								<div class=" mb-2.5 text-sm font-medium">OpenAI API Key</div>
								<div class="flex w-full">
									<div class="flex-1">
										<input
											class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
											placeholder="Enter OpenAI API Key"
											bind:value={OPENAI_API_KEY}
											autocomplete="off"
										/>
									</div>
								</div>
								<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
									Adds optional support for online models.
								</div>
							</div>

							<hr class=" dark:border-gray-700" />

							<div>
								<div class=" mb-2.5 text-sm font-medium">OpenAI API Base URL</div>
								<div class="flex w-full">
									<div class="flex-1">
										<input
											class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
											placeholder="Enter OpenAI API Key"
											bind:value={OPENAI_API_BASE_URL}
											autocomplete="off"
										/>
									</div>
								</div>
								<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
									WebUI will make requests to <span class=" text-gray-200"
										>'{OPENAI_API_BASE_URL}/chat'</span
									>
								</div>
							</div>
						</div>

						<div class="flex justify-end pt-3 text-sm font-medium">
							<button
								class=" px-4 py-2 bg-emerald-600 hover:bg-emerald-700 text-gray-100 transition rounded"
								type="submit"
							>
								Save
							</button>
						</div>
					</form>
1431
				{:else if selectedTab === 'addons'}
Timothy J. Baek's avatar
Timothy J. Baek committed
1432
1433
1434
1435
1436
					<form
						class="flex flex-col h-full justify-between space-y-3 text-sm"
						on:submit|preventDefault={() => {
							saveSettings({
								gravatarEmail: gravatarEmail !== '' ? gravatarEmail : undefined,
1437
								gravatarUrl: gravatarEmail !== '' ? getGravatarURL(gravatarEmail) : undefined
Timothy J. Baek's avatar
Timothy J. Baek committed
1438
1439
1440
1441
							});
							show = false;
						}}
					>
1442
						<div class=" space-y-3">
1443
							<div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1444
1445
1446
1447
1448
								<div class=" mb-1 text-sm font-medium">WebUI Add-ons</div>

								<div>
									<div class=" py-0.5 flex w-full justify-between">
										<div class=" self-center text-xs font-medium">Title Auto Generation</div>
1449

Timothy J. Baek's avatar
Timothy J. Baek committed
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
										<button
											class="p-1 px-3 text-xs flex rounded transition"
											on:click={() => {
												toggleTitleAutoGenerate();
											}}
											type="button"
										>
											{#if titleAutoGenerate === true}
												<span class="ml-2 self-center">On</span>
											{:else}
												<span class="ml-2 self-center">Off</span>
											{/if}
										</button>
									</div>
								</div>
1465

Timothy J. Baek's avatar
Timothy J. Baek committed
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
								<div>
									<div class=" py-0.5 flex w-full justify-between">
										<div class=" self-center text-xs font-medium">Voice Input Auto-Send</div>

										<button
											class="p-1 px-3 text-xs flex rounded transition"
											on:click={() => {
												toggleSpeechAutoSend();
											}}
											type="button"
										>
											{#if speechAutoSend === true}
												<span class="ml-2 self-center">On</span>
											{:else}
												<span class="ml-2 self-center">Off</span>
											{/if}
										</button>
									</div>
1484
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506

								<div>
									<div class=" py-0.5 flex w-full justify-between">
										<div class=" self-center text-xs font-medium">
											Response AutoCopy to Clipboard
										</div>

										<button
											class="p-1 px-3 text-xs flex rounded transition"
											on:click={() => {
												toggleResponseAutoCopy();
											}}
											type="button"
										>
											{#if responseAutoCopy === true}
												<span class="ml-2 self-center">On</span>
											{:else}
												<span class="ml-2 self-center">Off</span>
											{/if}
										</button>
									</div>
								</div>
1507
1508
1509
							</div>

							<hr class=" dark:border-gray-700" />
Timothy J. Baek's avatar
Timothy J. Baek committed
1510
1511
1512
1513
1514
1515
1516
							<div>
								<div class=" mb-2.5 text-sm font-medium">
									Gravatar Email <span class=" text-gray-400 text-sm">(optional)</span>
								</div>
								<div class="flex w-full">
									<div class="flex-1">
										<input
Timothy J. Baek's avatar
Timothy J. Baek committed
1517
											class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
Timothy J. Baek's avatar
Timothy J. Baek committed
1518
1519
1520
1521
1522
1523
1524
											placeholder="Enter Your Email"
											bind:value={gravatarEmail}
											autocomplete="off"
											type="email"
										/>
									</div>
								</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1525
								<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
Timothy J. Baek's avatar
Timothy J. Baek committed
1526
									Changes user profile image to match your <a
Timothy J. Baek's avatar
Timothy J. Baek committed
1527
										class=" text-gray-500 dark:text-gray-300 font-medium"
Timothy J. Baek's avatar
Timothy J. Baek committed
1528
1529
1530
1531
1532
										href="https://gravatar.com/"
										target="_blank">Gravatar.</a
									>
								</div>
							</div>
1533
1534
						</div>

Timothy J. Baek's avatar
Timothy J. Baek committed
1535
1536
1537
1538
1539
1540
1541
1542
1543
						<div class="flex justify-end pt-3 text-sm font-medium">
							<button
								class=" px-4 py-2 bg-emerald-600 hover:bg-emerald-700 text-gray-100 transition rounded"
								type="submit"
							>
								Save
							</button>
						</div>
					</form>
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686
1687
				{:else if selectedTab === 'chats'}
					<div class="flex flex-col h-full justify-between space-y-3 text-sm">
						<div class="flex flex-col">
							<input
								id="chat-import-input"
								bind:files={importFiles}
								type="file"
								accept=".json"
								hidden
							/>
							<button
								class=" flex rounded-md py-2 px-3.5 w-full hover:bg-gray-200 dark:hover:bg-gray-800 transition"
								on:click={() => {
									document.getElementById('chat-import-input').click();
								}}
							>
								<div class=" self-center mr-3">
									<svg
										xmlns="http://www.w3.org/2000/svg"
										viewBox="0 0 16 16"
										fill="currentColor"
										class="w-4 h-4"
									>
										<path
											fill-rule="evenodd"
											d="M4 2a1.5 1.5 0 0 0-1.5 1.5v9A1.5 1.5 0 0 0 4 14h8a1.5 1.5 0 0 0 1.5-1.5V6.621a1.5 1.5 0 0 0-.44-1.06L9.94 2.439A1.5 1.5 0 0 0 8.878 2H4Zm4 9.5a.75.75 0 0 1-.75-.75V8.06l-.72.72a.75.75 0 0 1-1.06-1.06l2-2a.75.75 0 0 1 1.06 0l2 2a.75.75 0 1 1-1.06 1.06l-.72-.72v2.69a.75.75 0 0 1-.75.75Z"
											clip-rule="evenodd"
										/>
									</svg>
								</div>
								<div class=" self-center text-sm font-medium">Import Chats</div>
							</button>
							<button
								class=" flex rounded-md py-2 px-3.5 w-full hover:bg-gray-200 dark:hover:bg-gray-800 transition"
								on:click={() => {
									exportChats();
								}}
							>
								<div class=" self-center mr-3">
									<svg
										xmlns="http://www.w3.org/2000/svg"
										viewBox="0 0 16 16"
										fill="currentColor"
										class="w-4 h-4"
									>
										<path
											fill-rule="evenodd"
											d="M4 2a1.5 1.5 0 0 0-1.5 1.5v9A1.5 1.5 0 0 0 4 14h8a1.5 1.5 0 0 0 1.5-1.5V6.621a1.5 1.5 0 0 0-.44-1.06L9.94 2.439A1.5 1.5 0 0 0 8.878 2H4Zm4 3.5a.75.75 0 0 1 .75.75v2.69l.72-.72a.75.75 0 1 1 1.06 1.06l-2 2a.75.75 0 0 1-1.06 0l-2-2a.75.75 0 0 1 1.06-1.06l.72.72V6.25A.75.75 0 0 1 8 5.5Z"
											clip-rule="evenodd"
										/>
									</svg>
								</div>
								<div class=" self-center text-sm font-medium">Export Chats</div>
							</button>
						</div>
						<!-- {#if showDeleteHistoryConfirm}
							<div
								class="flex justify-between rounded-md items-center py-3 px-3.5 w-full transition"
							>
								<div class="flex items-center">
									<svg
										xmlns="http://www.w3.org/2000/svg"
										fill="none"
										viewBox="0 0 24 24"
										stroke-width="1.5"
										stroke="currentColor"
										class="w-5 h-5 mr-3"
									>
										<path
											stroke-linecap="round"
											stroke-linejoin="round"
											d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0"
										/>
									</svg>
									<span>Are you sure?</span>
								</div>

								<div class="flex space-x-1.5 items-center">
									<button
										class="hover:text-white transition"
										on:click={() => {
											deleteChatHistory();
											showDeleteHistoryConfirm = false;
										}}
									>
										<svg
											xmlns="http://www.w3.org/2000/svg"
											viewBox="0 0 20 20"
											fill="currentColor"
											class="w-4 h-4"
										>
											<path
												fill-rule="evenodd"
												d="M16.704 4.153a.75.75 0 01.143 1.052l-8 10.5a.75.75 0 01-1.127.075l-4.5-4.5a.75.75 0 011.06-1.06l3.894 3.893 7.48-9.817a.75.75 0 011.05-.143z"
												clip-rule="evenodd"
											/>
										</svg>
									</button>
									<button
										class="hover:text-white transition"
										on:click={() => {
											showDeleteHistoryConfirm = false;
										}}
									>
										<svg
											xmlns="http://www.w3.org/2000/svg"
											viewBox="0 0 20 20"
											fill="currentColor"
											class="w-4 h-4"
										>
											<path
												d="M6.28 5.22a.75.75 0 00-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 101.06 1.06L10 11.06l3.72 3.72a.75.75 0 101.06-1.06L11.06 10l3.72-3.72a.75.75 0 00-1.06-1.06L10 8.94 6.28 5.22z"
											/>
										</svg>
									</button>
								</div>
							</div>
						{:else}
							<button
								class=" flex rounded-md py-3 px-3.5 w-full hover:bg-gray-900 transition"
								on:click={() => {
									showDeleteHistoryConfirm = true;
								}}
							>
								<div class="mr-3">
									<svg
										xmlns="http://www.w3.org/2000/svg"
										fill="none"
										viewBox="0 0 24 24"
										stroke-width="1.5"
										stroke="currentColor"
										class="w-5 h-5"
									>
										<path
											stroke-linecap="round"
											stroke-linejoin="round"
											d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0"
										/>
									</svg>
								</div>
								<span>Clear conversations</span>
							</button>
						{/if} -->
					</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1688
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
1717
1718
1719
1720
1721
1722
1723
1724
1725
1726
1727
1728
1729
1730
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
1788
1789
1790
1791
1792
1793
1794
1795
				{:else if selectedTab === 'auth'}
					<form
						class="flex flex-col h-full justify-between space-y-3 text-sm"
						on:submit|preventDefault={() => {
							console.log('auth save');
							saveSettings({
								authHeader: authEnabled ? `${authType} ${authContent}` : undefined
							});
							show = false;
						}}
					>
						<div class=" space-y-3">
							<div>
								<div class=" py-1 flex w-full justify-between">
									<div class=" self-center text-sm font-medium">Authorization Header</div>

									<button
										class="p-1 px-3 text-xs flex rounded transition"
										type="button"
										on:click={() => {
											toggleAuthHeader();
										}}
									>
										{#if authEnabled === true}
											<svg
												xmlns="http://www.w3.org/2000/svg"
												viewBox="0 0 24 24"
												fill="currentColor"
												class="w-4 h-4"
											>
												<path
													fill-rule="evenodd"
													d="M12 1.5a5.25 5.25 0 00-5.25 5.25v3a3 3 0 00-3 3v6.75a3 3 0 003 3h10.5a3 3 0 003-3v-6.75a3 3 0 00-3-3v-3c0-2.9-2.35-5.25-5.25-5.25zm3.75 8.25v-3a3.75 3.75 0 10-7.5 0v3h7.5z"
													clip-rule="evenodd"
												/>
											</svg>

											<span class="ml-2 self-center"> On </span>
										{:else}
											<svg
												xmlns="http://www.w3.org/2000/svg"
												viewBox="0 0 24 24"
												fill="currentColor"
												class="w-4 h-4"
											>
												<path
													d="M18 1.5c2.9 0 5.25 2.35 5.25 5.25v3.75a.75.75 0 01-1.5 0V6.75a3.75 3.75 0 10-7.5 0v3a3 3 0 013 3v6.75a3 3 0 01-3 3H3.75a3 3 0 01-3-3v-6.75a3 3 0 013-3h9v-3c0-2.9 2.35-5.25 5.25-5.25z"
												/>
											</svg>

											<span class="ml-2 self-center">Off</span>
										{/if}
									</button>
								</div>
							</div>

							{#if authEnabled}
								<hr class=" dark:border-gray-700" />

								<div class="mt-2">
									<div class=" py-1 flex w-full space-x-2">
										<button
											class=" py-1 font-semibold flex rounded transition"
											on:click={() => {
												authType = authType === 'Basic' ? 'Bearer' : 'Basic';
											}}
											type="button"
										>
											{#if authType === 'Basic'}
												<span class="self-center mr-2">Basic</span>
											{:else if authType === 'Bearer'}
												<span class="self-center mr-2">Bearer</span>
											{/if}
										</button>

										<div class="flex-1">
											<input
												class="w-full rounded py-2 px-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none"
												placeholder="Enter Authorization Header Content"
												bind:value={authContent}
											/>
										</div>
									</div>
									<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
										Toggle between <span class=" text-gray-500 dark:text-gray-300 font-medium"
											>'Basic'</span
										>
										and <span class=" text-gray-500 dark:text-gray-300 font-medium">'Bearer'</span> by
										clicking on the label next to the input.
									</div>
								</div>

								<hr class=" dark:border-gray-700" />

								<div>
									<div class=" mb-2.5 text-sm font-medium">Preview Authorization Header</div>
									<textarea
										value={JSON.stringify({
											Authorization: `${authType} ${authContent}`
										})}
										class="w-full rounded p-4 text-sm dark:text-gray-300 dark:bg-gray-800 outline-none resize-none"
										rows="2"
										disabled
									/>
								</div>
							{/if}
						</div>

1796
1797
						<div class="flex justify-end pt-3 text-sm font-medium">
							<button
Timothy J. Baek's avatar
Timothy J. Baek committed
1798
								class=" px-4 py-2 bg-emerald-600 hover:bg-emerald-700 text-gray-100 transition rounded"
Timothy J. Baek's avatar
Timothy J. Baek committed
1799
								type="submit"
1800
1801
1802
1803
							>
								Save
							</button>
						</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1804
					</form>
Timothy J. Baek's avatar
Timothy J. Baek committed
1805
				{:else if selectedTab === 'about'}
Timothy J. Baek's avatar
Timothy J. Baek committed
1806
					<div class="flex flex-col h-full justify-between space-y-3 text-sm mb-6">
Timothy J. Baek's avatar
Timothy J. Baek committed
1807
1808
1809
1810
						<div class=" space-y-3">
							<div>
								<div class=" mb-2.5 text-sm font-medium">Ollama Web UI Version</div>
								<div class="flex w-full">
Timothy J. Baek's avatar
Timothy J. Baek committed
1811
									<div class="flex-1 text-xs text-gray-700 dark:text-gray-200">
1812
										{$config && $config.version ? $config.version : WEB_UI_VERSION}
Timothy J. Baek's avatar
Timothy J. Baek committed
1813
									</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1814
1815
1816
1817
1818
								</div>
							</div>

							<hr class=" dark:border-gray-700" />

Timothy J. Baek's avatar
Timothy J. Baek committed
1819
1820
1821
1822
							<div>
								<div class=" mb-2.5 text-sm font-medium">Ollama Version</div>
								<div class="flex w-full">
									<div class="flex-1 text-xs text-gray-700 dark:text-gray-200">
Timothy J. Baek's avatar
Timothy J. Baek committed
1823
										{ollamaVersion ?? 'N/A'}
Timothy J. Baek's avatar
Timothy J. Baek committed
1824
1825
1826
1827
1828
1829
									</div>
								</div>
							</div>

							<hr class=" dark:border-gray-700" />

Timothy J. Baek's avatar
Timothy J. Baek committed
1830
1831
1832
1833
1834
1835
1836
1837
1838
							<div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
								Created by <a
									class=" text-gray-500 dark:text-gray-300 font-medium"
									href="https://github.com/tjbck"
									target="_blank">Timothy J. Baek</a
								>
							</div>

							<div>
1839
1840
								<a href="https://github.com/ollama-webui/ollama-webui">
									<img
1841
										alt="Github Repo"
1842
1843
1844
										src="https://img.shields.io/github/stars/ollama-webui/ollama-webui?style=social&label=Star us on Github"
									/>
								</a>
Timothy J. Baek's avatar
Timothy J. Baek committed
1845
1846
1847
							</div>
						</div>
					</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1848
1849
				{/if}
			</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1850
		</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
1851
1852
	</div>
</Modal>
1853
1854
1855
1856
1857
1858
1859
1860
1861

<style>
	input::-webkit-outer-spin-button,
	input::-webkit-inner-spin-button {
		/* display: none; <- Crashes Chrome on hover */
		-webkit-appearance: none;
		margin: 0; /* <-- Apparently some margin are still there even though it's hidden */
	}

1862
1863
1864
1865
1866
1867
1868
1869
1870
	.tabs::-webkit-scrollbar {
		display: none; /* for Chrome, Safari and Opera */
	}

	.tabs {
		-ms-overflow-style: none; /* IE and Edge */
		scrollbar-width: none; /* Firefox */
	}

1871
1872
1873
1874
	input[type='number'] {
		-moz-appearance: textfield; /* Firefox */
	}
</style>