2022-03-03 00:42:43 +08:00
|
|
|
<script lang="ts">
|
2023-02-24 05:32:18 +08:00
|
|
|
import DropdownOptions from "./DropdownOptions.svelte";
|
2023-01-24 02:41:33 +08:00
|
|
|
import { createEventDispatcher } from "svelte";
|
2022-06-02 01:02:18 +08:00
|
|
|
import { BlockTitle } from "@gradio/atoms";
|
2023-02-24 05:32:18 +08:00
|
|
|
import { Remove, DropdownArrow } from "@gradio/icons";
|
2022-03-03 00:42:43 +08:00
|
|
|
export let label: string;
|
2023-02-23 07:16:15 +08:00
|
|
|
export let info: string | undefined = undefined;
|
2023-03-01 02:18:33 +08:00
|
|
|
export let value: string | Array<string> | undefined;
|
2023-01-05 08:13:46 +08:00
|
|
|
export let multiselect: boolean = false;
|
2023-02-24 05:32:18 +08:00
|
|
|
export let max_choices: number;
|
2022-03-03 00:42:43 +08:00
|
|
|
export let choices: Array<string>;
|
2022-04-08 03:36:49 +08:00
|
|
|
export let disabled: boolean = false;
|
2022-04-27 18:47:15 +08:00
|
|
|
export let show_label: boolean;
|
2023-01-24 02:41:33 +08:00
|
|
|
|
|
|
|
const dispatch = createEventDispatcher<{
|
2023-02-24 05:32:18 +08:00
|
|
|
change: string | Array<string> | undefined;
|
2023-01-24 02:41:33 +08:00
|
|
|
}>();
|
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
let inputValue: string,
|
|
|
|
activeOption: string,
|
|
|
|
showOptions = false;
|
|
|
|
|
|
|
|
$: filtered = choices.filter((o) =>
|
|
|
|
inputValue ? o.toLowerCase().includes(inputValue.toLowerCase()) : o
|
|
|
|
);
|
|
|
|
$: if (
|
|
|
|
(activeOption && !filtered.includes(activeOption)) ||
|
|
|
|
(!activeOption && inputValue)
|
|
|
|
)
|
|
|
|
activeOption = filtered[0];
|
|
|
|
|
|
|
|
$: readonly =
|
|
|
|
(!multiselect && typeof value === "string") ||
|
|
|
|
(multiselect && Array.isArray(value) && value.length === max_choices);
|
|
|
|
|
2023-02-23 23:55:43 +08:00
|
|
|
// The initial value of value is [] so that can
|
|
|
|
// cause infinite loops in the non-multiselect case
|
|
|
|
$: if (!multiselect && !Array.isArray(value)) {
|
2023-01-24 02:41:33 +08:00
|
|
|
dispatch("change", value);
|
|
|
|
}
|
2023-02-24 05:32:18 +08:00
|
|
|
|
|
|
|
function add(option: string) {
|
|
|
|
if (Array.isArray(value)) {
|
2023-03-01 02:18:33 +08:00
|
|
|
if (!max_choices || value.length < max_choices) {
|
2023-02-24 05:32:18 +08:00
|
|
|
value.push(option);
|
|
|
|
dispatch("change", value);
|
|
|
|
}
|
|
|
|
showOptions = !(value.length === max_choices);
|
|
|
|
}
|
|
|
|
value = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
function remove(option: string) {
|
|
|
|
if (Array.isArray(value)) {
|
|
|
|
value = value.filter((v: string) => v !== option);
|
|
|
|
dispatch("change", value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function remove_all(e: any) {
|
2023-03-01 02:18:33 +08:00
|
|
|
if (multiselect) {
|
|
|
|
value = [];
|
|
|
|
} else {
|
|
|
|
value = "";
|
|
|
|
}
|
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
inputValue = "";
|
|
|
|
e.preventDefault();
|
|
|
|
dispatch("change", value);
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleOptionMousedown(e: any) {
|
|
|
|
const option = e.detail.target.dataset.value;
|
|
|
|
inputValue = "";
|
|
|
|
|
|
|
|
if (option !== undefined) {
|
|
|
|
if (!multiselect) {
|
|
|
|
value = option;
|
|
|
|
inputValue = "";
|
|
|
|
dispatch("change", value);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (value?.includes(option)) {
|
|
|
|
remove(option);
|
|
|
|
} else {
|
|
|
|
add(option);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleKeyup(e: any) {
|
|
|
|
if (e.key === "Enter" && activeOption != undefined) {
|
|
|
|
if (!multiselect) {
|
|
|
|
value = activeOption;
|
|
|
|
inputValue = "";
|
|
|
|
} else if (multiselect && Array.isArray(value)) {
|
|
|
|
value.includes(activeOption) ? remove(activeOption) : add(activeOption);
|
|
|
|
inputValue = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (e.key === "ArrowUp" || e.key === "ArrowDown") {
|
|
|
|
const increment = e.key === "ArrowUp" ? -1 : 1;
|
|
|
|
const calcIndex = filtered.indexOf(activeOption) + increment;
|
|
|
|
activeOption =
|
|
|
|
calcIndex < 0
|
|
|
|
? filtered[filtered.length - 1]
|
|
|
|
: calcIndex === filtered.length
|
|
|
|
? filtered[0]
|
|
|
|
: filtered[calcIndex];
|
|
|
|
}
|
|
|
|
if (e.key === "Escape") {
|
|
|
|
showOptions = false;
|
|
|
|
}
|
|
|
|
}
|
2022-03-03 00:42:43 +08:00
|
|
|
</script>
|
|
|
|
|
2023-01-05 08:13:46 +08:00
|
|
|
<!-- svelte-ignore a11y-label-has-associated-control -->
|
2022-04-26 22:48:39 +08:00
|
|
|
<label>
|
2023-02-23 07:16:15 +08:00
|
|
|
<BlockTitle {show_label} {info}>{label}</BlockTitle>
|
2023-01-18 04:47:40 +08:00
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
<div class="wrap">
|
|
|
|
<div class="wrap-inner" class:showOptions>
|
|
|
|
{#if Array.isArray(value)}
|
|
|
|
{#each value as s}
|
|
|
|
<div on:click|preventDefault={() => remove(s)} class="token">
|
|
|
|
<span>{s}</span>
|
|
|
|
<div
|
|
|
|
class:hidden={disabled}
|
|
|
|
class="token-remove"
|
|
|
|
title="Remove {s}"
|
|
|
|
>
|
|
|
|
<Remove />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{/each}
|
|
|
|
{:else}
|
|
|
|
<span class="single-select">{value}</span>
|
|
|
|
{/if}
|
|
|
|
<div class="secondary-wrap">
|
|
|
|
<input
|
|
|
|
class="border-none"
|
|
|
|
{disabled}
|
|
|
|
{readonly}
|
|
|
|
autocomplete="off"
|
|
|
|
bind:value={inputValue}
|
|
|
|
on:focus={() =>
|
|
|
|
(showOptions =
|
|
|
|
Array.isArray(value) && value.length === max_choices
|
|
|
|
? false
|
|
|
|
: true)}
|
|
|
|
on:blur={() => (showOptions = false)}
|
|
|
|
on:keyup={handleKeyup}
|
|
|
|
/>
|
|
|
|
<div
|
|
|
|
class:hide={!value?.length || disabled}
|
|
|
|
class="token-remove remove-all"
|
|
|
|
title="Remove All"
|
|
|
|
on:click={remove_all}
|
|
|
|
>
|
|
|
|
<Remove />
|
|
|
|
</div>
|
|
|
|
<DropdownArrow />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<DropdownOptions
|
|
|
|
bind:value
|
|
|
|
{showOptions}
|
|
|
|
{filtered}
|
|
|
|
{activeOption}
|
|
|
|
{disabled}
|
|
|
|
on:change={handleOptionMousedown}
|
|
|
|
/>
|
|
|
|
</div>
|
2022-04-26 22:48:39 +08:00
|
|
|
</label>
|
2023-01-18 04:47:40 +08:00
|
|
|
|
|
|
|
<style>
|
2023-02-24 05:32:18 +08:00
|
|
|
.wrap {
|
2023-01-18 04:47:40 +08:00
|
|
|
--ring-color: transparent;
|
|
|
|
position: relative;
|
2023-01-21 05:24:24 +08:00
|
|
|
box-shadow: 0 0 0 var(--shadow-spread) var(--ring-color),
|
2023-01-25 03:51:51 +08:00
|
|
|
var(--shadow-inset);
|
2023-02-24 05:32:18 +08:00
|
|
|
border: 1px solid var(--color-border-primary);
|
2023-01-18 04:47:40 +08:00
|
|
|
border-radius: var(--radius-lg);
|
|
|
|
}
|
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
.wrap:focus-within {
|
2023-01-18 04:47:40 +08:00
|
|
|
--ring-color: var(--color-focus-ring);
|
|
|
|
border-color: var(--input-border-color-focus);
|
|
|
|
}
|
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
.wrap-inner {
|
|
|
|
display: flex;
|
|
|
|
position: relative;
|
|
|
|
flex-wrap: wrap;
|
|
|
|
align-items: center;
|
2023-01-18 04:47:40 +08:00
|
|
|
}
|
|
|
|
|
2023-02-24 05:32:18 +08:00
|
|
|
.token {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
cursor: pointer;
|
|
|
|
margin: var(--size-1);
|
|
|
|
box-shadow: var(--shadow-drop);
|
|
|
|
border: 1px solid var(--checkbox-label-border-color-base);
|
|
|
|
border-radius: var(--radius-md);
|
|
|
|
background: var(--checkbox-label-background-base);
|
|
|
|
padding: var(--size-1-5) var(--size-3);
|
|
|
|
color: var(--color-text-body);
|
|
|
|
font-size: var(--scale-00);
|
|
|
|
line-height: var(--line-md);
|
|
|
|
}
|
|
|
|
|
|
|
|
.token > * + * {
|
|
|
|
margin-left: var(--size-2);
|
|
|
|
}
|
|
|
|
|
|
|
|
.token:hover {
|
|
|
|
border: 1px solid var(--icon_button-border-color-hover);
|
|
|
|
color: var(--color-text-label);
|
|
|
|
}
|
|
|
|
|
|
|
|
.token-remove {
|
|
|
|
fill: var(--color-text-body);
|
|
|
|
display: flex;
|
|
|
|
justify-content: center;
|
|
|
|
align-items: center;
|
|
|
|
cursor: pointer;
|
|
|
|
border: 1px solid var(--color-border-primary);
|
|
|
|
border-radius: var(--radius-full);
|
|
|
|
background: var(--color-background-tertiary);
|
|
|
|
padding: var(--size-0-5);
|
|
|
|
width: 18px;
|
|
|
|
height: 18px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.token-remove:hover,
|
|
|
|
.remove-all:hover {
|
|
|
|
border: 1px solid var(--icon_button-border-color-hover);
|
|
|
|
color: var(--color-text-label);
|
|
|
|
}
|
|
|
|
|
|
|
|
.single-select {
|
|
|
|
margin: var(--size-2);
|
|
|
|
color: var(--color-text-body);
|
|
|
|
}
|
|
|
|
|
|
|
|
.secondary-wrap {
|
|
|
|
display: flex;
|
|
|
|
flex: 1 1 0%;
|
|
|
|
align-items: center;
|
|
|
|
border: none;
|
|
|
|
min-width: min-content;
|
|
|
|
}
|
|
|
|
|
|
|
|
input {
|
|
|
|
outline: none;
|
|
|
|
border: none;
|
|
|
|
background: inherit;
|
|
|
|
padding: var(--size-2-5);
|
|
|
|
width: 100%;
|
|
|
|
color: var(--color-text-body);
|
|
|
|
font-size: var(--scale-00);
|
|
|
|
}
|
|
|
|
|
|
|
|
input:disabled {
|
2023-01-18 04:47:40 +08:00
|
|
|
cursor: not-allowed;
|
|
|
|
box-shadow: none;
|
|
|
|
}
|
2023-02-24 05:32:18 +08:00
|
|
|
|
|
|
|
.remove-all {
|
|
|
|
margin-left: var(--size-1);
|
|
|
|
width: 20px;
|
|
|
|
height: 20px;
|
|
|
|
}
|
2023-01-18 04:47:40 +08:00
|
|
|
</style>
|