|
<script lang="ts"> |
|
import { getContext } from 'svelte'; |
|
import CitationsModal from './CitationsModal.svelte'; |
|
import Collapsible from '$lib/components/common/Collapsible.svelte'; |
|
import ChevronDown from '$lib/components/icons/ChevronDown.svelte'; |
|
import ChevronUp from '$lib/components/icons/ChevronUp.svelte'; |
|
|
|
const i18n = getContext('i18n'); |
|
|
|
export let id = ''; |
|
export let sources = []; |
|
|
|
let citations = []; |
|
let showPercentage = false; |
|
let showRelevance = true; |
|
|
|
let showCitationModal = false; |
|
let selectedCitation: any = null; |
|
let isCollapsibleOpen = false; |
|
|
|
function calculateShowRelevance(sources: any[]) { |
|
const distances = sources.flatMap((citation) => citation.distances ?? []); |
|
const inRange = distances.filter((d) => d !== undefined && d >= -1 && d <= 1).length; |
|
const outOfRange = distances.filter((d) => d !== undefined && (d < -1 || d > 1)).length; |
|
|
|
if (distances.length === 0) { |
|
return false; |
|
} |
|
|
|
if ( |
|
(inRange === distances.length - 1 && outOfRange === 1) || |
|
(outOfRange === distances.length - 1 && inRange === 1) |
|
) { |
|
return false; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
function shouldShowPercentage(sources: any[]) { |
|
const distances = sources.flatMap((citation) => citation.distances ?? []); |
|
return distances.every((d) => d !== undefined && d >= -1 && d <= 1); |
|
} |
|
|
|
$: { |
|
console.log('sources', sources); |
|
citations = sources.reduce((acc, source) => { |
|
if (Object.keys(source).length === 0) { |
|
return acc; |
|
} |
|
|
|
source.document.forEach((document, index) => { |
|
const metadata = source.metadata?.[index]; |
|
const distance = source.distances?.[index]; |
|
|
|
|
|
const id = metadata?.source ?? source?.source?.id ?? 'N/A'; |
|
let _source = source?.source; |
|
|
|
if (metadata?.name) { |
|
_source = { ..._source, name: metadata.name }; |
|
} |
|
|
|
if (id.startsWith('http://') || id.startsWith('https://')) { |
|
_source = { ..._source, name: id, url: id }; |
|
} |
|
|
|
const existingSource = acc.find((item) => item.id === id); |
|
|
|
if (existingSource) { |
|
existingSource.document.push(document); |
|
existingSource.metadata.push(metadata); |
|
if (distance !== undefined) existingSource.distances.push(distance); |
|
} else { |
|
acc.push({ |
|
id: id, |
|
source: _source, |
|
document: [document], |
|
metadata: metadata ? [metadata] : [], |
|
distances: distance !== undefined ? [distance] : undefined |
|
}); |
|
} |
|
}); |
|
return acc; |
|
}, []); |
|
|
|
showRelevance = calculateShowRelevance(citations); |
|
showPercentage = shouldShowPercentage(citations); |
|
} |
|
</script> |
|
|
|
<CitationsModal |
|
bind:show={showCitationModal} |
|
citation={selectedCitation} |
|
{showPercentage} |
|
{showRelevance} |
|
/> |
|
|
|
{#if citations.length > 0} |
|
<div class=" py-0.5 -mx-0.5 w-full flex gap-1 items-center flex-wrap"> |
|
{#if citations.length <= 3} |
|
<div class="flex text-xs font-medium flex-wrap"> |
|
{#each citations as citation, idx} |
|
<button |
|
id={`source-${id}-${idx}`} |
|
class="no-toggle outline-hidden flex dark:text-gray-300 p-1 bg-white dark:bg-gray-900 rounded-xl max-w-96" |
|
on:click={() => { |
|
showCitationModal = true; |
|
selectedCitation = citation; |
|
}} |
|
> |
|
{#if citations.every((c) => c.distances !== undefined)} |
|
<div class="bg-gray-50 dark:bg-gray-800 rounded-full size-4"> |
|
{idx + 1} |
|
</div> |
|
{/if} |
|
<div |
|
class="flex-1 mx-1 truncate text-black/60 hover:text-black dark:text-white/60 dark:hover:text-white transition" |
|
> |
|
{citation.source.name} |
|
</div> |
|
</button> |
|
{/each} |
|
</div> |
|
{:else} |
|
<Collapsible |
|
id="collapsible-sources" |
|
bind:open={isCollapsibleOpen} |
|
className="w-full max-w-full " |
|
buttonClassName="w-fit max-w-full" |
|
> |
|
<div |
|
class="flex w-full overflow-auto items-center gap-2 text-gray-500 hover:text-gray-600 dark:hover:text-gray-400 transition cursor-pointer" |
|
> |
|
<div |
|
class="flex-1 flex items-center gap-1 overflow-auto scrollbar-none w-full max-w-full" |
|
> |
|
<span class="whitespace-nowrap hidden sm:inline shrink-0" |
|
>{$i18n.t('References from')}</span |
|
> |
|
<div class="flex items-center overflow-auto scrollbar-none w-full max-w-full flex-1"> |
|
<div class="flex text-xs font-medium items-center"> |
|
{#each citations.slice(0, 2) as citation, idx} |
|
<button |
|
class="no-toggle outline-hidden flex dark:text-gray-300 p-1 bg-gray-50 hover:bg-gray-100 dark:bg-gray-900 dark:hover:bg-gray-850 transition rounded-xl max-w-96" |
|
on:click={() => { |
|
showCitationModal = true; |
|
selectedCitation = citation; |
|
}} |
|
on:pointerup={(e) => { |
|
e.stopPropagation(); |
|
}} |
|
> |
|
{#if citations.every((c) => c.distances !== undefined)} |
|
<div class="bg-gray-50 dark:bg-gray-800 rounded-full size-4"> |
|
{idx + 1} |
|
</div> |
|
{/if} |
|
<div class="flex-1 mx-1 truncate"> |
|
{citation.source.name} |
|
</div> |
|
</button> |
|
{/each} |
|
</div> |
|
</div> |
|
<div class="flex items-center gap-1 whitespace-nowrap shrink-0"> |
|
<span class="hidden sm:inline">{$i18n.t('and')}</span> |
|
{citations.length - 2} |
|
<span>{$i18n.t('more')}</span> |
|
</div> |
|
</div> |
|
<div class="shrink-0"> |
|
{#if isCollapsibleOpen} |
|
<ChevronUp strokeWidth="3.5" className="size-3.5" /> |
|
{:else} |
|
<ChevronDown strokeWidth="3.5" className="size-3.5" /> |
|
{/if} |
|
</div> |
|
</div> |
|
<div slot="content"> |
|
<div class="flex text-xs font-medium flex-wrap"> |
|
{#each citations as citation, idx} |
|
<button |
|
id={`source-${id}-${idx}`} |
|
class="no-toggle outline-hidden flex dark:text-gray-300 p-1 bg-gray-50 hover:bg-gray-100 dark:bg-gray-900 dark:hover:bg-gray-850 transition rounded-xl max-w-96" |
|
on:click={() => { |
|
showCitationModal = true; |
|
selectedCitation = citation; |
|
}} |
|
> |
|
{#if citations.every((c) => c.distances !== undefined)} |
|
<div class="bg-gray-50 dark:bg-gray-800 rounded-full size-4"> |
|
{idx + 1} |
|
</div> |
|
{/if} |
|
<div class="flex-1 mx-1 truncate"> |
|
{citation.source.name} |
|
</div> |
|
</button> |
|
{/each} |
|
</div> |
|
</div> |
|
</Collapsible> |
|
{/if} |
|
</div> |
|
{/if} |
|
|