| <script lang="ts"> |
| import { getContext } from 'svelte'; |
| const i18n = getContext('i18n'); |
| |
| import { WEBUI_API_BASE_URL, WEBUI_BASE_URL } from '$lib/constants'; |
| |
| import ChevronDown from '$lib/components/icons/ChevronDown.svelte'; |
| import Clipboard from '$lib/components/icons/Clipboard.svelte'; |
| import GarbageBin from '$lib/components/icons/GarbageBin.svelte'; |
| import Tooltip from '$lib/components/common/Tooltip.svelte'; |
| |
| import { toast } from 'svelte-sonner'; |
| import dayjs from 'dayjs'; |
| |
| export let webhook; |
| export let expanded = false; |
| |
| export let onClick = () => {}; |
| export let onDelete = () => {}; |
| export let onUpdate = (changes: { name: string; profile_image_url: string }) => {}; |
| |
| let name = webhook.name; |
| let image = webhook.profile_image_url || ''; |
| |
| |
| $: if (name !== webhook.name || image !== (webhook.profile_image_url || '')) { |
| onUpdate({ name: name.trim() || webhook.name, profile_image_url: image }); |
| } |
| |
| let filesInputElement; |
| let inputFiles; |
| |
| const handleImageUpload = () => { |
| if (!inputFiles?.length) return; |
| |
| const reader = new FileReader(); |
| reader.onload = (event) => { |
| const dataUrl = `${event.target?.result}`; |
| const fileType = inputFiles[0]?.type; |
| |
| if (['image/gif', 'image/webp'].includes(fileType)) { |
| image = dataUrl; |
| } else { |
| const tempImage = new Image(); |
| tempImage.src = dataUrl; |
| tempImage.onload = () => { |
| const canvas = document.createElement('canvas'); |
| const canvasSize = 100; |
| canvas.width = canvasSize; |
| canvas.height = canvasSize; |
| |
| const context = canvas.getContext('2d'); |
| const aspectRatio = tempImage.width / tempImage.height; |
| const scaledWidth = aspectRatio > 1 ? canvasSize * aspectRatio : canvasSize; |
| const scaledHeight = aspectRatio > 1 ? canvasSize : canvasSize / aspectRatio; |
| const offsetX = (canvasSize - scaledWidth) / 2; |
| const offsetY = (canvasSize - scaledHeight) / 2; |
| |
| context.drawImage(tempImage, offsetX, offsetY, scaledWidth, scaledHeight); |
| image = canvas.toDataURL('image/webp', 0.8); |
| }; |
| } |
| inputFiles = null; |
| }; |
| reader.readAsDataURL(inputFiles[0]); |
| }; |
| |
| const copyUrl = () => { |
| navigator.clipboard.writeText( |
| `${WEBUI_API_BASE_URL}/channels/webhooks/${webhook.id}/${webhook.token}` |
| ); |
| toast.success($i18n.t('Copied')); |
| }; |
| </script> |
|
|
| <input |
| bind:this={filesInputElement} |
| bind:files={inputFiles} |
| type="file" |
| hidden |
| accept="image/*" |
| on:change={handleImageUpload} |
| /> |
|
|
| <div class="text-xs -mx-1"> |
| |
| <button |
| type="button" |
| class="w-full flex items-center gap-3 px-3.5 py-3 hover:bg-gray-50 dark:hover:bg-gray-900 rounded-xl transition" |
| on:click={onClick} |
| > |
| <img |
| src={image || `${WEBUI_BASE_URL}/static/favicon.png`} |
| class="rounded-full size-8 object-cover flex-shrink-0" |
| alt="" |
| /> |
| <div class="flex-1 text-left min-w-0"> |
| <div class="font-medium text-gray-900 dark:text-white truncate"> |
| {name} |
| </div> |
| <div class="text-gray-500 text-xs"> |
| {$i18n.t('Created on {{date}}', { |
| date: dayjs(webhook.created_at / 1000000).format('MMM D, YYYY') |
| })} |
| {#if webhook.user?.name} |
| {$i18n.t('by {{name}}', { name: webhook.user.name })} |
| {/if} |
| </div> |
| </div> |
| <ChevronDown |
| className="size-3.5 text-gray-400 transition-transform duration-200 {expanded |
| ? 'rotate-180' |
| : ''}" |
| /> |
| </button> |
|
|
| |
| {#if expanded} |
| <div class="mt-1 mb-3 px-3.5 py-3 border border-gray-100 dark:border-gray-850 rounded-2xl"> |
| <div class="flex items-center gap-3"> |
| <button |
| type="button" |
| class="shrink-0 rounded-xl overflow-hidden hover:opacity-80 transition" |
| on:click={() => filesInputElement.click()} |
| > |
| <img |
| src={image || `${WEBUI_BASE_URL}/static/favicon.png`} |
| class="size-8 object-cover" |
| alt="" |
| /> |
| </button> |
| <div class="flex-1"> |
| <div class=" text-gray-500 text-xs">{$i18n.t('Name')}</div> |
| <input |
| type="text" |
| class="w-full text-sm bg-transparent outline-none placeholder:text-gray-300 dark:placeholder:text-gray-700" |
| bind:value={name} |
| placeholder={$i18n.t('Webhook Name')} |
| /> |
| </div> |
| <div class="flex items-center gap-1"> |
| <Tooltip content={$i18n.t('Copy URL')}> |
| <button |
| type="button" |
| class="p-1.5 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-800 transition" |
| on:click={copyUrl} |
| > |
| <Clipboard className="size-4 text-gray-500" /> |
| </button> |
| </Tooltip> |
| <Tooltip content={$i18n.t('Delete')}> |
| <button |
| type="button" |
| class="p-1.5 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-800 transition" |
| on:click={onDelete} |
| > |
| <GarbageBin className="size-4 text-gray-500" /> |
| </button> |
| </Tooltip> |
| </div> |
| </div> |
| </div> |
| {/if} |
| </div> |
|
|