<script lang="ts"> import { createEventDispatcher } from "svelte"; import IconGear from "~icons/bi/gear-fill"; import { base } from "$app/paths"; import type { Assistant } from "$lib/types/Assistant"; import { formatUserCount } from "$lib/utils/formatUserCount"; import IconInternet from "../icons/IconInternet.svelte"; import CarbonExport from "~icons/carbon/export"; import CarbonCheckmark from "~icons/carbon/checkmark"; import CarbonUserMultiple from "~icons/carbon/user-multiple"; import { share } from "$lib/utils/share"; import { PUBLIC_ORIGIN, PUBLIC_SHARE_PREFIX } from "$env/static/public"; import { page } from "$app/stores"; export let assistant: Pick< Assistant, | "avatar" | "name" | "rag" | "dynamicPrompt" | "modelId" | "createdByName" | "exampleInputs" | "_id" | "description" | "userCount" >; const dispatch = createEventDispatcher<{ message: string }>(); $: hasRag = assistant?.rag?.allowAllDomains || (assistant?.rag?.allowedDomains?.length ?? 0) > 0 || (assistant?.rag?.allowedLinks?.length ?? 0) > 0 || assistant?.dynamicPrompt; const prefix = PUBLIC_SHARE_PREFIX || `${PUBLIC_ORIGIN || $page.url.origin}${base}`; $: shareUrl = `${prefix}/assistant/${assistant?._id}`; let isCopied = false; </script> <div class="flex h-full w-full flex-col content-center items-center justify-center pb-52"> <div class="relative mt-auto rounded-2xl bg-gray-100 text-gray-600 dark:border-gray-800 dark:bg-gray-800/60 dark:text-gray-300" > <div class="mt-3 flex min-w-[80dvw] items-center gap-4 p-4 pr-1 sm:min-w-[440px] md:p-8 xl:gap-8" > {#if assistant.avatar} <img src={`${base}/settings/assistants/${assistant._id.toString()}/avatar.jpg?hash=${ assistant.avatar }`} alt="avatar" class="size-16 flex-none rounded-full object-cover max-sm:self-start md:size-32" /> {:else} <div class="flex size-12 flex-none items-center justify-center rounded-full bg-gray-300 object-cover text-xl font-bold uppercase text-gray-500 max-sm:self-start sm:text-4xl md:size-32 dark:bg-gray-600" > {assistant?.name[0]} </div> {/if} <div class="flex h-full flex-col gap-2 text-balance"> <p class="-mb-1">Assistant</p> <p class="text-xl font-bold sm:text-2xl">{assistant.name}</p> {#if assistant.description} <p class="line-clamp-6 text-sm text-gray-500 dark:text-gray-400"> {assistant.description} </p> {/if} {#if hasRag} <div class="flex h-5 w-fit items-center gap-1 rounded-full bg-blue-500/10 pl-1 pr-2 text-xs" title="This assistant uses the websearch." > <IconInternet classNames="text-sm text-blue-600" /> Has internet access </div> {/if} {#if assistant.createdByName} <p class="pt-1 text-sm text-gray-400 dark:text-gray-500"> Created by <a class="hover:underline" href="{base}/assistants?user={assistant.createdByName}"> {assistant.createdByName} </a> {#if assistant.userCount && assistant.userCount > 1} <span class="mx-1">ยท</span> <div class="inline-flex items-baseline gap-1 text-sm text-gray-400 dark:text-gray-500" title="Number of users" > <CarbonUserMultiple class="text-xxs" />{formatUserCount(assistant.userCount)} users </div> {/if} </p> {/if} </div> </div> <div class="absolute right-3 top-3 md:right-4 md:top-4"> <div class="flex flex-row items-center gap-1"> <button class="flex h-7 items-center gap-1.5 rounded-full border bg-white px-2.5 py-1 text-gray-800 shadow-sm hover:shadow-inner max-sm:px-1.5 md:text-sm dark:border-gray-700 dark:bg-gray-700 dark:text-gray-300/90 dark:hover:bg-gray-800" on:click={() => { if (!isCopied) { share(shareUrl, assistant.name); isCopied = true; setTimeout(() => { isCopied = false; }, 2000); } }} > {#if isCopied} <CarbonCheckmark class="text-xxs text-green-600 max-sm:text-xs" /> <span class="text-green-600 max-sm:hidden"> Link copied </span> {:else} <CarbonExport class="text-xxs max-sm:text-xs" /> <span class="max-sm:hidden"> Share </span> {/if} </button> <a href="{base}/settings/assistants/{assistant._id.toString()}" class="flex h-7 items-center gap-1.5 rounded-full border bg-white px-2.5 py-1 text-gray-800 shadow-sm hover:shadow-inner md:text-sm dark:border-gray-700 dark:bg-gray-700 dark:text-gray-300/90 dark:hover:bg-gray-800" ><IconGear class="text-xxs" />Settings</a > </div> </div> </div> {#if assistant.exampleInputs} <div class="mx-auto mt-auto w-full gap-8 sm:-mb-8"> <div class="md:col-span-2 md:mt-6"> <div class="grid grid-cols-1 gap-3 {assistant.exampleInputs.length > 1 ? 'md:grid-cols-2' : ''}" > {#each assistant.exampleInputs as example} <button type="button" class="truncate whitespace-nowrap rounded-xl border bg-gray-50 px-3 py-2 text-left text-smd text-gray-600 hover:bg-gray-100 dark:border-gray-800 dark:bg-gray-800 dark:text-gray-300 dark:hover:bg-gray-700" on:click={() => dispatch("message", example)} > {example} </button> {/each} </div> </div> </div> {/if} </div>