File size: 6,720 Bytes
cafb235
 
 
5b796dc
 
cafb235
5b796dc
124338b
5b796dc
3d58f23
5ab594c
 
5b796dc
124338b
3bb3aef
5ab594c
 
7d7a53f
31daf3d
e1f5db2
7d7a53f
 
 
cafb235
21b8785
 
 
7d7a53f
21b8785
 
 
 
 
 
 
 
 
 
 
 
 
 
 
cafb235
 
3d58f23
21b8785
3d58f23
21b8785
 
 
 
5ab594c
0e5ff83
31daf3d
5ab594c
21b8785
5ab594c
21b8785
5b796dc
 
cafb235
 
e8df589
 
606d8b6
 
 
 
 
 
 
 
 
 
 
3bb3aef
606d8b6
3bb3aef
606d8b6
3bb3aef
 
606d8b6
 
 
 
e8df589
606d8b6
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
e8df589
606d8b6
 
 
 
 
3d58f23
606d8b6
 
 
 
 
 
 
 
cafb235
606d8b6
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
124338b
606d8b6
 
 
 
 
 
 
 
 
cafb235
606d8b6
 
 
 
 
 
 
 
 
 
 
3d58f23
606d8b6
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
5ab594c
606d8b6
5ab594c
cafb235
 
606d8b6
e8df589
606d8b6
 
 
 
 
 
 
 
 
 
 
 
 
cafb235
 
 
606d8b6
cafb235
1
2
3
4
5
6
7
8
9
10
11
12
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
53
54
55
56
57
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
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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
<script lang="ts">
	import { createEventDispatcher } from "svelte";
	import { base } from "$app/paths";
	import { goto } from "$app/navigation";
	import type { Model } from "$lib/types/Model";
	import type { Assistant } from "$lib/types/Assistant";
	import { useSettingsStore } from "$lib/stores/settings";
	import { formatUserCount } from "$lib/utils/formatUserCount";
	import IconGear from "~icons/bi/gear-fill";
	import IconInternet from "../icons/IconInternet.svelte";
	import CarbonExport from "~icons/carbon/export";
	import CarbonCheckmark from "~icons/carbon/checkmark";
	import CarbonRenew from "~icons/carbon/renew";
	import CarbonUserMultiple from "~icons/carbon/user-multiple";
	import CarbonTools from "~icons/carbon/tools";

	import { share } from "$lib/utils/share";
	import { usePublicConfig } from "$lib/utils/PublicConfig.svelte";

	import { page } from "$app/state";
	import type { Serialize } from "$lib/utils/serialize";

	const publicConfig = usePublicConfig();

	interface Props {
		models: Model[];
		assistant: Pick<
			Serialize<Assistant>,
			| "avatar"
			| "name"
			| "rag"
			| "dynamicPrompt"
			| "modelId"
			| "createdByName"
			| "exampleInputs"
			| "_id"
			| "description"
			| "userCount"
			| "tools"
		>;
	}

	let { models, assistant }: Props = $props();

	const dispatch = createEventDispatcher<{ message: string }>();

	let hasRag = $derived(
		assistant?.rag?.allowAllDomains ||
			(assistant?.rag?.allowedDomains?.length ?? 0) > 0 ||
			(assistant?.rag?.allowedLinks?.length ?? 0) > 0 ||
			assistant?.dynamicPrompt
	);

	const prefix =
		publicConfig.PUBLIC_SHARE_PREFIX || `${publicConfig.PUBLIC_ORIGIN || page.url.origin}${base}`;

	let shareUrl = $derived(`${prefix}/assistant/${assistant?._id}`);

	let isCopied = $state(false);

	const settings = useSettingsStore();
</script>

<div class="my-auto grid gap-8 lg:grid-cols-9">
	<div class="lg:col-span-4">
		<div>
			<div class="mb-3 flex items-center">
				{#if assistant.avatar}
					<img
						src={`${base}/settings/assistants/${assistant._id.toString()}/avatar.jpg?hash=${
							assistant.avatar
						}`}
						alt="avatar"
						class="mr-3 size-10 flex-none rounded-full object-cover"
					/>
				{:else}
					<div
						class="mr-3 flex size-10 flex-none items-center justify-center rounded-full bg-gray-300 object-cover text-xl font-bold uppercase text-gray-500 dark:bg-gray-600"
					>
						{assistant?.name[0]}
					</div>
				{/if}
				<div class="text-2xl font-semibold">
					{assistant.name}
				</div>
			</div>
			<p class="line-clamp-5 text-sm leading-relaxed text-gray-600 dark:text-gray-400">
				{assistant.description || "No description provided."}
			</p>

			<button
				onclick={() => {
					settings.instantSet({
						activeModel: models[0].name,
					});
					goto(`${base}/`);
				}}
				class="mt-4 inline-flex w-fit items-center rounded-md border border-gray-200 bg-gray-50 px-2 py-1 text-xs text-gray-500 hover:bg-gray-100 dark:border-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:hover:bg-gray-700"
			>
				<CarbonRenew class="mr-1.5 text-xxs" /> Reset to default model
			</button>
		</div>
	</div>
	<div class="lg:col-span-5 lg:pl-12">
		<div class="overflow-hidden rounded-xl border dark:border-gray-800">
			<div class="flex flex-wrap items-center justify-between gap-2 p-3">
				<div class="flex flex-wrap items-center gap-2">
					<div class="hidden text-sm font-medium text-gray-600 dark:text-gray-400 sm:block">
						About this Assistant
					</div>
					{#if assistant.createdByName}
						<span class="hidden text-gray-400 sm:block"></span>
						<a
							class="text-sm text-gray-500 hover:underline"
							href="{base}/assistants?user={assistant.createdByName}"
						>
							{#if !import.meta.env.SSR && window.innerWidth < 640}By
							{/if}{assistant.createdByName}
						</a>
					{/if}
				</div>
				<div class="flex gap-1 self-start">
					<button
						class="btn flex h-7 w-7 rounded-full bg-gray-100 p-1 text-xs hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-800 dark:hover:bg-gray-600"
						onclick={() => {
							if (!isCopied) {
								share(shareUrl, assistant.name);
								isCopied = true;
								setTimeout(() => {
									isCopied = false;
								}, 2000);
							}
						}}
						title="Share assistant"
					>
						{#if isCopied}
							<CarbonCheckmark class="text-green-600" />
						{:else}
							<CarbonExport />
						{/if}
					</button>
					<a
						href="{base}/settings/assistants/{assistant._id.toString()}"
						aria-label="Settings"
						title="Settings"
						class="btn flex h-7 w-7 rounded-full bg-gray-100 p-1 text-xs hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-800 dark:hover:bg-gray-600"
						><IconGear /></a
					>
				</div>
			</div>
			<div class="grid gap-3 bg-gray-50 p-3 text-sm dark:bg-gray-800/70">
				<div class="flex flex-wrap gap-2">
					{#if hasRag}
						<div
							class="flex h-6 items-center gap-1 rounded-full bg-blue-500/10 pl-1.5 pr-2.5 text-xs"
							title="This assistant uses web search"
						>
							<IconInternet classNames="text-sm text-blue-600" />
							Internet access
						</div>
					{/if}

					{#if assistant?.tools?.length}
						<div
							class="flex h-6 items-center gap-1 rounded-full bg-purple-500/10 pl-1.5 pr-2.5 text-xs"
							title="This assistant can use tools"
						>
							<CarbonTools class="text-sm text-purple-600" />
							Has tools
						</div>
					{/if}

					{#if assistant.userCount && assistant.userCount > 1}
						<div
							class="flex h-6 items-center gap-1 rounded-full bg-gray-500/10 pl-1.5 pr-2.5 text-xs"
							title="Number of users"
						>
							<CarbonUserMultiple class="text-sm text-gray-600 dark:text-gray-400" />
							{formatUserCount(assistant.userCount)} users
						</div>
					{/if}
				</div>
			</div>
		</div>
	</div>
	{#if assistant.exampleInputs && assistant.exampleInputs.length > 0}
		<div class="lg:col-span-9 lg:mt-6">
			<p class="mb-3 text-center text-gray-600 dark:text-gray-300 lg:text-left">Examples</p>
			<div
				class="flex max-h-60 gap-2 overflow-x-auto pb-2 text-center scrollbar-thin scrollbar-thumb-gray-300 dark:scrollbar-thumb-gray-700 lg:grid lg:grid-cols-3 lg:overflow-y-auto lg:text-left"
			>
				{#each assistant.exampleInputs as example}
					<button
						type="button"
						class="flex-shrink-0 rounded-xl border bg-gray-50 px-2.5 py-2 text-sm text-gray-600 hover:bg-gray-100 dark:border-gray-800 dark:bg-gray-800 dark:text-gray-300 dark:hover:bg-gray-700 sm:px-3 lg:w-full xl:px-3.5 xl:text-base"
						onclick={() => dispatch("message", example)}
					>
						{example}
					</button>
				{/each}
			</div>
		</div>
	{/if}
	<div class="h-40 sm:h-24"></div>
</div>