<script lang="ts">
  import type { Label, ModelData } from '$lib'
  import { AuthorBrief, DATASETS, Icon } from '$lib'
  import pkg from '$site/package.json'
  import { format_num } from 'matterviz'
  import { tooltip } from 'svelte-multiselect/attachments'
  import type { HTMLAttributes } from 'svelte/elements'
  import { fade, slide } from 'svelte/transition'

  interface Props extends HTMLAttributes<HTMLElementTagNameMap[`section`]> {
    model: ModelData
    metrics: readonly Label[]
    sort_by: keyof ModelData
    show_details?: boolean
    metrics_style?: string
    title_style?: string
  }
  let {
    model,
    metrics,
    sort_by,
    show_details = $bindable(false),
    metrics_style = ``,
    title_style = ``,
    ...rest
  }: Props = $props()

  let { model_name, model_key, model_params, training_set } = $derived(model)
  let all_metrics = $derived({
    ...(typeof model.metrics?.discovery === `object`
      ? model.metrics.discovery.unique_prototypes
      : {}),
    ...(typeof model.metrics?.phonons == `object`
      ? model.metrics?.phonons.kappa_103
      : {}),
    CPS: model.CPS,
  })
  let { missing_preds } = $derived(all_metrics)

  let links = $derived(
    [
      [model.repo, `Repo`, `GitHub`],
      [model.paper, `Paper`, `Paper`],
      [model.url, `Docs`, `Docs`],
      [model.checkpoint_url, `Checkpoint`, `Download`],
      [`${pkg.repository}/blob/-/models/${model.dirname}`, `Files`, `Directory`],
    ] as const,
  )
  const target = { target: `_blank`, rel: `noopener` }
  let n_model_params = $derived(format_num(model_params, `.3~s`))
  let expand_title = $derived(
    `${show_details ? `Hide` : `Show`} authors and package versions`,
  )
</script>

<h2 style={title_style}>
  <a href="/models/{model_key}">{model_name}</a>
  <button
    onclick={() => (show_details = !show_details)}
    aria-expanded={show_details}
    aria-label={expand_title}
    title={expand_title}
    style={title_style}
  >
    <Icon icon="Arrow{show_details ? `Up` : `Down`}" />
  </button>
</h2>
<nav>
  {#each links.filter(([href]) => href?.startsWith(`http`)) as
    [href, title, icon]
    (title)
  }
    <span>
      <Icon {icon} />
      <a {href} {...target}>{title}</a>
    </span>
  {/each}
</nav>

<section class="metadata" {...rest}>
  {#if training_set}
    <span style="grid-column: span 2">
      <Icon icon="Database" />
      Training data:
      {#each training_set as train_set_key, idx (train_set_key)}
        {#if idx > 0}
          &nbsp;+&nbsp;
        {/if}
        {@const training_set = DATASETS[train_set_key]}
        {@const { n_structures, name, slug, n_materials } = training_set}
        {@const pretty_n_mat = typeof n_materials == `number`
        ? format_num(n_materials)
        : n_materials}
        {@const n_mat_str = n_materials ? ` from ${pretty_n_mat} materials` : ``}
        <a
          href="/data/{slug}"
          title="{name}: {format_num(n_structures)} structures{n_mat_str}"
          {@attach tooltip()}
        >{train_set_key}</a>
      {/each}
    </span>
  {/if}
  <span title="Date added">
    <Icon icon="Calendar" />
    Added {model.date_added}
  </span>
  {#if model.date_published}
    <span title="Date published">
      <Icon icon="CalendarCheck" />
      Published {model.date_published}
    </span>
  {/if}
  <span>
    <Icon icon="NeuralNetwork" /> {n_model_params} params
  </span>
  {#if model.n_estimators > 1}
    <span>
      <Icon icon="Forest" />
      Ensemble of {model.n_estimators}
      <span
        title="This result used a model ensemble with {model.n_estimators} members with {n_model_params} parameters each."
        {@attach tooltip()}
      >
        &nbsp;<Icon icon="Info" />
      </span>
    </span>
  {/if}
  <span
    {@attach tooltip()}
    title="Out of {format_num(DATASETS.WBM.n_structures, `,`)} WBM structures, {format_num(missing_preds ?? 0, `,`)} are missing predictions. This refers only to the discovery task of predicting WBM convex hull distances."
  >
    <Icon icon="MissingMetadata" /> Missing preds:
    {format_num(missing_preds ?? 0, `,.0f`)}
    {#if missing_preds && missing_preds > 0}
      <small>({format_num(missing_preds / DATASETS.WBM.n_structures, `.3~%`)})</small>
    {/if}
  </span>
</section>
{#if show_details}
  <div transition:fade|fly={{ duration: 200 }}>
    <section transition:slide={{ duration: 200 }}>
      <h3>Authors</h3>
      <ul>
        {#each model.authors as author (author.name)}
          <li><AuthorBrief {author} /></li>
        {/each}
      </ul>
      {#if model.trained_by}
        <h3>Trained By</h3>
        <ul>
          {#each model.trained_by as author (author.name)}
            <li><AuthorBrief {author} /></li>
          {/each}
        </ul>
      {/if}
    </section>
    <section transition:slide={{ duration: 200 }}>
      <h3>Package versions</h3>
      <ul>
        {#each Object.entries(model.requirements ?? {}) as [name, version] (name)}
          {@const [href, link_text] = version?.startsWith(`http`)
          // version.split(`/`).at(-1) assumes final part after / of URL is the package version, as is the case for GitHub releases
          ? [version, version.split(`/`).at(-1)]
          : [`https://pypi.org/project/${name}/${version}`, version]}
          <li style="font-size: smaller">
            {name}: <a {href} {...target}>{link_text}</a>
          </li>
        {/each}
      </ul>
    </section>
  </div>
{/if}

<section class="metrics" style={metrics_style || null}>
  <h3 style="margin: 0; font-weight: normal">Metrics</h3>
  <ul>
    <!-- hide run time if value is 0 (i.e. not available) -->
    {#each metrics as metric (JSON.stringify(metric))}
      {@const { key, label, short, unit } = metric}
      {@const value = all_metrics[key as keyof typeof all_metrics] as number}
      <li class:active={sort_by == key}>
        <label for={key}>{@html short ?? label ?? key}</label>
        <strong>{isNaN(value) ? `n/a` : format_num(value)}
          <small>{unit ?? ``}</small></strong>
      </li>
    {/each}
  </ul>
</section>

<style>
  h2 {
    margin: 8pt 0 0;
    text-align: center;
    border-radius: 5pt;
  }
  h2 a {
    color: inherit;
  }
  button {
    background: none;
    padding: 0;
    font: inherit;
  }
  ul {
    list-style: disc;
  }
  nav {
    font-weight: 250;
    display: flex;
    gap: 5pt 1em;
    flex-wrap: wrap;
    place-content: center;
  }
  nav > span {
    display: flex;
    gap: 6pt;
    place-items: center;
  }
  section.metadata {
    display: grid;
    gap: 9pt 5pt;
    grid-template-columns: 1fr 1fr;
    font-size: 0.95em;
    align-content: center;
  }
  div {
    display: grid;
    grid-template-columns: 1fr 1fr;
    gap: 15pt;
    justify-content: space-between;
  }
  small {
    font-weight: 100;
    font-size: 8pt;
  }
  section.metrics > ul {
    display: grid;
    grid-template-columns: repeat(auto-fill, minmax(9em, 1fr));
    gap: 3pt 1em;
    list-style: none;
    max-height: 10em;
    padding: 0;
  }
  section.metrics > ul > li {
    font-weight: lighter;
    display: flex;
    justify-content: space-between;
  }
  section.metrics > ul > li > :is(label, strong) {
    padding: 0 4pt;
    border-radius: 3pt;
  }
  section.metrics > ul > li.active > label {
    font-weight: bold;
  }
  /* prevent long from increasing ModelCard container width */
  :is(section, div, nav) {
    word-break: break-word;
  }
</style>
