2023-08-30 18:11:01 +08:00
|
|
|
import { test, describe, assert, afterEach, vi } from "vitest";
|
2023-09-14 03:42:30 +08:00
|
|
|
import { cleanup, render } from "@gradio/tootils";
|
2023-08-25 01:22:41 +08:00
|
|
|
import event from "@testing-library/user-event";
|
|
|
|
import { setupi18n } from "../app/src/i18n";
|
|
|
|
|
2023-10-31 12:46:02 +08:00
|
|
|
import Dropdown from "./Index.svelte";
|
2023-08-25 01:22:41 +08:00
|
|
|
import type { LoadingStatus } from "@gradio/statustracker";
|
|
|
|
|
|
|
|
const loading_status: LoadingStatus = {
|
|
|
|
eta: 0,
|
|
|
|
queue_position: 1,
|
|
|
|
queue_size: 1,
|
|
|
|
status: "complete" as LoadingStatus["status"],
|
|
|
|
scroll_to_output: false,
|
|
|
|
visible: true,
|
|
|
|
fn_index: 0,
|
|
|
|
show_progress: "full"
|
|
|
|
};
|
|
|
|
|
|
|
|
describe("Dropdown", () => {
|
2023-08-30 18:11:01 +08:00
|
|
|
afterEach(() => {
|
|
|
|
cleanup();
|
|
|
|
vi.useRealTimers();
|
|
|
|
});
|
2023-08-25 01:22:41 +08:00
|
|
|
beforeEach(() => {
|
|
|
|
setupi18n();
|
|
|
|
});
|
|
|
|
test("renders provided value", async () => {
|
|
|
|
const { getByLabelText } = await render(Dropdown, {
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
2023-09-13 00:58:18 +08:00
|
|
|
max_choices: null,
|
2023-08-25 01:22:41 +08:00
|
|
|
value: "choice",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["choice", "choice"],
|
|
|
|
["choice2", "choice2"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: false,
|
|
|
|
interactive: false
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
assert.equal(item.value, "choice");
|
|
|
|
});
|
|
|
|
|
|
|
|
test("selecting the textbox should show the options", async () => {
|
2023-09-13 00:58:18 +08:00
|
|
|
const { getByLabelText, getAllByTestId } = await render(Dropdown, {
|
2023-08-25 01:22:41 +08:00
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
max_choices: 10,
|
|
|
|
value: "choice",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["choice", "choice"],
|
|
|
|
["name2", "choice2"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
|
|
|
await item.focus();
|
|
|
|
|
|
|
|
const options = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options).toHaveLength(2);
|
|
|
|
expect(options[0]).toContainHTML("choice");
|
2023-09-13 00:58:18 +08:00
|
|
|
expect(options[1]).toContainHTML("name2");
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
test("editing the textbox value should filter the options", async () => {
|
2023-09-13 00:58:18 +08:00
|
|
|
const { getByLabelText, getAllByTestId } = await render(Dropdown, {
|
2023-08-25 01:22:41 +08:00
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
max_choices: 10,
|
|
|
|
value: "",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
|
|
|
await item.focus();
|
|
|
|
const options = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options).toHaveLength(2);
|
|
|
|
|
2023-09-13 00:58:18 +08:00
|
|
|
item.value = "";
|
2023-08-25 01:22:41 +08:00
|
|
|
await event.keyboard("z");
|
|
|
|
const options_new = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options_new).toHaveLength(1);
|
|
|
|
expect(options[0]).toContainHTML("zebra");
|
|
|
|
});
|
|
|
|
|
2023-08-30 18:11:01 +08:00
|
|
|
test("blurring the textbox should cancel the filter", async () => {
|
|
|
|
const { getByLabelText, listen } = await render(Dropdown, {
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
value: "default",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
max_choices: undefined,
|
|
|
|
choices: [
|
|
|
|
["default", "default"],
|
|
|
|
["other", "other"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: false,
|
|
|
|
interactive: true
|
2023-08-30 18:11:01 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
const change_event = listen("change");
|
|
|
|
const select_event = listen("select");
|
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
item.focus();
|
2023-08-30 18:11:01 +08:00
|
|
|
await event.keyboard("other");
|
2023-09-14 03:42:30 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
test("blurring the textbox should save the input value", async () => {
|
|
|
|
const { getByLabelText, listen } = await render(Dropdown, {
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
value: "default",
|
|
|
|
label: "Dropdown",
|
|
|
|
max_choices: undefined,
|
|
|
|
allow_custom_value: true,
|
|
|
|
choices: [
|
|
|
|
["dwight", "dwight"],
|
|
|
|
["michael", "michael"]
|
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-09-14 03:42:30 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
const change_event = listen("change");
|
|
|
|
|
|
|
|
item.focus();
|
|
|
|
await event.keyboard("kevin");
|
2023-08-30 18:11:01 +08:00
|
|
|
await item.blur();
|
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
assert.equal(item.value, "kevin");
|
|
|
|
assert.equal(change_event.callCount, 1);
|
2023-08-30 18:11:01 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
test("focusing the label should toggle the options", async () => {
|
|
|
|
const { getByLabelText, listen } = await render(Dropdown, {
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
value: "default",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["default", "default"],
|
|
|
|
["other", "other"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-08-30 18:11:01 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
const blur_event = listen("blur");
|
|
|
|
const focus_event = listen("focus");
|
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
item.focus();
|
|
|
|
item.blur();
|
2023-08-30 18:11:01 +08:00
|
|
|
|
|
|
|
assert.equal(blur_event.callCount, 1);
|
|
|
|
assert.equal(focus_event.callCount, 1);
|
|
|
|
});
|
|
|
|
|
2023-08-25 01:22:41 +08:00
|
|
|
test("deselecting and reselcting a filtered dropdown should show all options again", async () => {
|
2023-08-30 18:11:01 +08:00
|
|
|
vi.useFakeTimers();
|
2023-09-13 00:58:18 +08:00
|
|
|
const { getByLabelText, getAllByTestId } = await render(Dropdown, {
|
2023-08-25 01:22:41 +08:00
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
max_choices: 10,
|
|
|
|
value: "",
|
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"],
|
|
|
|
["pony", "pony"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
item.focus();
|
2023-09-13 00:58:18 +08:00
|
|
|
item.value = "";
|
2023-08-25 01:22:41 +08:00
|
|
|
await event.keyboard("z");
|
|
|
|
const options = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options).toHaveLength(1);
|
|
|
|
|
|
|
|
await item.blur();
|
2023-08-30 18:11:01 +08:00
|
|
|
// Mock 100ms delay between interactions.
|
|
|
|
vi.runAllTimers();
|
2023-08-25 01:22:41 +08:00
|
|
|
await item.focus();
|
|
|
|
const options_new = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options_new).toHaveLength(3);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("passing in a new set of identical choices when the dropdown is open should not filter the dropdown", async () => {
|
|
|
|
const { getByLabelText, getAllByTestId, component } = await render(
|
|
|
|
Dropdown,
|
|
|
|
{
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
2023-09-13 00:58:18 +08:00
|
|
|
value: "",
|
2023-08-25 01:22:41 +08:00
|
|
|
label: "Dropdown",
|
2023-09-13 00:58:18 +08:00
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"],
|
|
|
|
["pony", "pony"]
|
2023-09-14 03:42:30 +08:00
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-08-25 01:22:41 +08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
|
|
|
await item.focus();
|
|
|
|
|
|
|
|
const options = getAllByTestId("dropdown-option");
|
|
|
|
|
|
|
|
expect(options).toHaveLength(3);
|
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
component.$set({
|
2023-09-13 00:58:18 +08:00
|
|
|
value: "",
|
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"],
|
|
|
|
["pony", "pony"]
|
|
|
|
]
|
|
|
|
});
|
2023-08-25 01:22:41 +08:00
|
|
|
|
2023-09-14 03:42:30 +08:00
|
|
|
item.focus();
|
2023-08-25 01:22:41 +08:00
|
|
|
|
2023-09-13 00:58:18 +08:00
|
|
|
const options_new = getAllByTestId("dropdown-option");
|
2023-08-25 01:22:41 +08:00
|
|
|
expect(options_new).toHaveLength(3);
|
|
|
|
});
|
2023-09-14 15:03:59 +08:00
|
|
|
|
|
|
|
test("setting a custom value when allow_custom_choice is false should revert to the first valid choice", async () => {
|
|
|
|
const { getByLabelText, getAllByTestId, component } = await render(
|
|
|
|
Dropdown,
|
|
|
|
{
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
value: "",
|
|
|
|
allow_custom_value: false,
|
|
|
|
label: "Dropdown",
|
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"],
|
|
|
|
["pony", "pony"]
|
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-09-14 15:03:59 +08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
|
|
|
await item.focus();
|
|
|
|
await event.keyboard("pie");
|
|
|
|
expect(item.value).toBe("applepie");
|
|
|
|
await item.blur();
|
|
|
|
expect(item.value).toBe("apple");
|
|
|
|
});
|
|
|
|
|
|
|
|
test("setting a custom value when allow_custom_choice is true should keep the value", async () => {
|
|
|
|
const { getByLabelText, getAllByTestId, component } = await render(
|
|
|
|
Dropdown,
|
|
|
|
{
|
|
|
|
show_label: true,
|
|
|
|
loading_status,
|
|
|
|
value: "",
|
|
|
|
allow_custom_value: true,
|
|
|
|
label: "Dropdown",
|
|
|
|
choices: [
|
|
|
|
["apple", "apple"],
|
|
|
|
["zebra", "zebra"],
|
|
|
|
["pony", "pony"]
|
|
|
|
],
|
2023-10-31 12:46:02 +08:00
|
|
|
filterable: true,
|
|
|
|
interactive: true
|
2023-09-14 15:03:59 +08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const item: HTMLInputElement = getByLabelText(
|
|
|
|
"Dropdown"
|
|
|
|
) as HTMLInputElement;
|
|
|
|
|
|
|
|
await item.focus();
|
|
|
|
await event.keyboard("pie");
|
|
|
|
expect(item.value).toBe("applepie");
|
|
|
|
await item.blur();
|
|
|
|
expect(item.value).toBe("applepie");
|
|
|
|
});
|
2023-08-25 01:22:41 +08:00
|
|
|
});
|