PromptCommands.svelte 2.99 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 { prompts } from '$lib/stores';
Timothy J. Baek's avatar
Timothy J. Baek committed
3
4
5
6
7
8
9
	import { findWordIndices } from '$lib/utils';
	import { tick } from 'svelte';

	export let prompt = '';
	let selectedCommandIdx = 0;
	let filteredPromptCommands = [];

Timothy J. Baek's avatar
Timothy J. Baek committed
10
	$: filteredPromptCommands = $prompts
11
12
		.filter((p) => p.command.includes(prompt))
		.sort((a, b) => a.title.localeCompare(b.title));
Timothy J. Baek's avatar
Timothy J. Baek committed
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52

	$: if (prompt) {
		selectedCommandIdx = 0;
	}

	export const selectUp = () => {
		selectedCommandIdx = Math.max(0, selectedCommandIdx - 1);
	};

	export const selectDown = () => {
		selectedCommandIdx = Math.min(selectedCommandIdx + 1, filteredPromptCommands.length - 1);
	};

	const confirmCommand = async (command) => {
		prompt = command.content;

		const chatInputElement = document.getElementById('chat-textarea');

		await tick();

		chatInputElement.style.height = '';
		chatInputElement.style.height = Math.min(chatInputElement.scrollHeight, 200) + 'px';

		chatInputElement?.focus();

		await tick();

		const words = findWordIndices(prompt);

		if (words.length > 0) {
			const word = words.at(0);
			chatInputElement.setSelectionRange(word?.startIndex, word.endIndex + 1);
		}
	};
</script>

{#if filteredPromptCommands.length > 0}
	<div class="md:px-2 mb-3 text-left w-full">
		<div class="flex w-full rounded-lg border border-gray-100 dark:border-gray-700">
			<div class=" bg-gray-100 dark:bg-gray-700 w-10 rounded-l-lg text-center">
Timothy J. Baek's avatar
Timothy J. Baek committed
53
				<div class=" text-lg font-semibold mt-2">/</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
54
			</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
55

Timothy J. Baek's avatar
Timothy J. Baek committed
56
			<div class="max-h-60 flex flex-col w-full rounded-r-lg">
Timothy J. Baek's avatar
Timothy J. Baek committed
57
				<div class=" overflow-y-auto bg-white p-2 rounded-tr-lg space-y-0.5">
Timothy J. Baek's avatar
Timothy J. Baek committed
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
					{#each filteredPromptCommands as command, commandIdx}
						<button
							class=" px-3 py-1.5 rounded-lg w-full text-left {commandIdx === selectedCommandIdx
								? ' bg-gray-100 selected-command-option-button'
								: ''}"
							type="button"
							on:click={() => {
								confirmCommand(command);
							}}
							on:mousemove={() => {
								selectedCommandIdx = commandIdx;
							}}
							on:focus={() => {}}
						>
							<div class=" font-medium text-black">
								{command.command}
							</div>

							<div class=" text-xs text-gray-600">
								{command.title}
							</div>
						</button>
					{/each}
				</div>

Timothy J. Baek's avatar
Timothy J. Baek committed
83
				<div
Timothy J. Baek's avatar
Timothy J. Baek committed
84
					class=" px-2 pb-1 text-xs text-gray-600 bg-white rounded-br-lg flex items-center space-x-1"
Timothy J. Baek's avatar
Timothy J. Baek committed
85
				>
Timothy J. Baek's avatar
Timothy J. Baek committed
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
					<div>
						<svg
							xmlns="http://www.w3.org/2000/svg"
							fill="none"
							viewBox="0 0 24 24"
							stroke-width="1.5"
							stroke="currentColor"
							class="w-3 h-3"
						>
							<path
								stroke-linecap="round"
								stroke-linejoin="round"
								d="m11.25 11.25.041-.02a.75.75 0 0 1 1.063.852l-.708 2.836a.75.75 0 0 0 1.063.853l.041-.021M21 12a9 9 0 1 1-18 0 9 9 0 0 1 18 0Zm-9-3.75h.008v.008H12V8.25Z"
							/>
						</svg>
					</div>

					<div class="line-clamp-1">
Timothy J. Baek's avatar
Timothy J. Baek committed
104
105
						Tip: Update multiple variable slots consecutively by pressing the tab key in the chat
						input after each replacement.
Timothy J. Baek's avatar
Timothy J. Baek committed
106
107
					</div>
				</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
108
109
110
111
			</div>
		</div>
	</div>
{/if}