<script lang="ts">
  import { HeatmapTable, Icon, IconList, TableControls } from '$lib'
  import { metric_better_as } from '$lib/metrics'
  import type {
    CellSnippetArgs,
    DiscoverySet,
    Label,
    LinkData,
    ModelData,
  } from '$lib/types'
  import type { Snippet } from 'svelte'
  import { click_outside } from 'svelte-multiselect/attachments'
  import type { HTMLAttributes } from 'svelte/elements'
  import { SvelteSet } from 'svelte/reactivity'
  import { ALL_METRICS, HYPERPARAMS, METADATA_COLS } from './labels'
  import { assemble_row_data } from './metrics'
  import { heatmap_class } from './table-export'

  interface Props extends HTMLAttributes<HTMLTableElement> {
    discovery_set?: DiscoverySet
    model_filter?: (model: ModelData) => boolean
    col_filter?: (col: Label) => boolean
    show_energy_only?: boolean
    show_non_compliant?: boolean
    show_heatmap?: boolean
    show_compliant?: boolean
    show_selected_only?: boolean
    active_files?: { name: string; url: string }[]
    active_model_name?: string
    pred_files_dropdown_pos?: { x: number; y: number; name: string } | null
    selected_models?: Set<string>
  }
  let {
    discovery_set = $bindable(`unique_prototypes`),
    model_filter = $bindable(() => true),
    col_filter = $bindable(() => true),
    show_energy_only = $bindable(false),
    show_non_compliant = $bindable(true),
    show_heatmap = $bindable(true),
    show_compliant = $bindable(true),
    show_selected_only = $bindable(false),
    active_files = $bindable([]),
    active_model_name = $bindable(``),
    pred_files_dropdown_pos = $bindable(null),
    selected_models = $bindable(new SvelteSet<string>()),
    ...rest
  }: Props = $props()

  const { model_name, training_set, targets, date_added, links } = METADATA_COLS
  const { checkpoint_license, code_license, org } = METADATA_COLS
  const { graph_construction_radius, model_params } = HYPERPARAMS

  let columns = $derived(
    [
      ...Object.values(ALL_METRICS),
      model_name,
      training_set,
      model_params,
      targets,
      date_added,
      links,
      graph_construction_radius,
      checkpoint_license,
      code_license,
      org,
    ]
      .map((col) => {
        const better = col.better ?? metric_better_as(col.label)

        // append better=higher/lower to tooltip if applicable
        let description = col.description ?? ``
        if (better === `higher` || better === `lower`) {
          description = description
            ? `${description} (${better}=better)`
            : `${better}=better`
        }
        const visible = col.visible !== false && col_filter(col)

        return { ...col, better, description, visible } as Label
      })
      // Ensure Model column comes first
      .sort((col1, _col2) => (col1.label === `Model` ? -1 : 1)),
  )

  let selected_count = $derived(selected_models.size)

  let metrics_data = $derived(
    assemble_row_data(
      discovery_set,
      model_filter,
      show_energy_only,
      show_non_compliant,
      show_compliant,
    )
      .filter((row) => {
        // If show_selected_only is true, only show selected models
        if (show_selected_only) {
          const model_name = String((row as ModelData).Model)
          return selected_models.has(model_name)
        }
        return true
      })
      .map((row) => {
        const model_data = row as ModelData
        const model_name = String(model_data.Model)
        const is_selected = selected_models.has(model_name)
        // Only apply selected styles when not filtering to show only selected models
        row.class = is_selected && !show_selected_only ? `highlight` : null
        return row
      }),
  )

  function show_dropdown(event: MouseEvent, links: LinkData) {
    event.stopPropagation()

    // Get button position for dropdown placement
    const button = event.currentTarget as HTMLElement
    const rect = button.getBoundingClientRect()

    active_model_name = links.pred_files.name
    active_files = links.pred_files.files

    // Position dropdown relative to the button's position in the document
    const { name } = links.pred_files
    pred_files_dropdown_pos = {
      x: rect.left + window.scrollX,
      y: rect.bottom + window.scrollY,
      name,
    }
  }
  const close_dropdown = () => (pred_files_dropdown_pos = null)

  function toggle_model_selection(model_name: string) {
    const new_selected = new SvelteSet(selected_models)
    if (new_selected.has(model_name)) new_selected.delete(model_name)
    else new_selected.add(model_name)
    selected_models = new_selected
  }

  function handle_row_double_click(event: MouseEvent, model_name: string) {
    event.preventDefault()
    toggle_model_selection(model_name)
  }
</script>

<svelte:window
  onkeydown={(event) => {
    if (event.key === `Escape` && pred_files_dropdown_pos) {
      close_dropdown()
      event.preventDefault()
    }
  }}
/>

{#snippet affiliation_cell({ row }: CellSnippetArgs)}
  <IconList icons={(row as ModelData).org_logos} />
{/snippet}

{#snippet links_cell({ val }: CellSnippetArgs)}
  {@const links = val as LinkData}
  {#each Object.entries(links).filter(([key]) => key !== `pred_files`) as
    [key, link]
    (JSON.stringify(link))
  }
    {#if `url` in link &&
    ![`missing`, `not available`, ``, null, undefined].includes(link.url)}
      <a href={link.url} target="_blank" rel="noopener noreferrer" title={link.title}>
        <Icon icon={link.icon} />
      </a>
    {:else}
      <span title="{key} not available">
        <Icon icon="Unavailable" />
      </span>
    {/if}
  {/each}
  {#if links?.pred_files}
    <button
      class="pred-files-btn"
      aria-label="Download model prediction files"
      onclick={(event) => show_dropdown(event, links)}
    >
      <Icon icon="Graph" />
    </button>
  {/if}
{/snippet}

<HeatmapTable
  data={metrics_data}
  {columns}
  initial_sort_column="CPS"
  initial_sort_direction="desc"
  sort_hint="Click on column headers to sort table rows"
  special_cells={{
    Links: links_cell as unknown as Snippet<[CellSnippetArgs]>,
    Org: affiliation_cell as unknown as Snippet<[CellSnippetArgs]>,
  }}
  default_num_format=".3f"
  bind:show_heatmap
  {heatmap_class}
  onrowdblclick={(event, row) => {
    const model_name = String((row as ModelData).Model)
    handle_row_double_click(event, model_name)
  }}
  {...rest}
>
  {#snippet controls()}
    <TableControls
      bind:show_energy_only
      bind:columns
      bind:show_heatmap
      bind:show_compliant
      bind:show_non_compliant
      bind:show_selected_only
      {selected_count}
    />
  {/snippet}
</HeatmapTable>

{#if pred_files_dropdown_pos}
  {@const { x, y } = pred_files_dropdown_pos}
  {@const style = `position: absolute; left: ${x}px; top: ${y}px;`}
  <div
    class="pred-files-dropdown"
    {style}
    {@attach click_outside({ callback: close_dropdown })}
  >
    <h4>Files for {active_model_name}</h4>
    <ol>
      {#each active_files as { name, url } (url)}
        <li>
          <a href={url} target="_blank" rel="noopener noreferrer">
            {@html name}
          </a>
        </li>
      {/each}
    </ol>
  </div>
{/if}

<style>
  .pred-files-btn {
    background: none;
    padding: 0;
  }
  .pred-files-dropdown {
    transform: translateX(-100%);
    margin-left: 20px;
    background: var(--page-bg);
    border: 1px solid var(--border);
    border-radius: 5px;
    padding: 4pt 11pt;
  }
  .pred-files-dropdown h4 {
    margin: 0;
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
  }
  .pred-files-dropdown ol {
    margin: 0;
    padding-left: 1em;
  }
</style>
