Add .key_up event listener to gr.Dropdown() (#7404)

* dropdown choice bug

* add changeset

* add changeset

* changes

* add changeset

* format

* key down:

* change

* change

* format

* add KeyDownData

* changes

* add demo

* notebook

* add changeset

* key up

* notebook

* changes

* Delete .changeset/shaggy-hairs-peel.md

* Delete .changeset/tasty-spies-spend.md

* revert changeset deletion

* change

* fix unit test

* type disable

* fix

* cset

* add changeset

---------

Co-authored-by: gradio-pr-bot <gradio-pr-bot@users.noreply.github.com>
This commit is contained in:
Abubakar Abid 2024-02-13 17:14:07 -08:00 committed by GitHub
parent 65437ce832
commit 065c5b163c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
12 changed files with 85 additions and 10 deletions

View File

@ -0,0 +1,8 @@
---
"@gradio/app": minor
"@gradio/dropdown": minor
"@gradio/utils": minor
"gradio": minor
---
fix:Add `.key_up` event listener to `gr.Dropdown()`

View File

@ -0,0 +1 @@
{"cells": [{"cell_type": "markdown", "id": "302934307671667531413257853548643485645", "metadata": {}, "source": ["# Gradio Demo: dropdown_key_up"]}, {"cell_type": "code", "execution_count": null, "id": "272996653310673477252411125948039410165", "metadata": {}, "outputs": [], "source": ["!pip install -q gradio "]}, {"cell_type": "code", "execution_count": null, "id": "288918539441861185822528903084949547379", "metadata": {}, "outputs": [], "source": ["import gradio as gr\n", "\n", "def test(value, key_up_data: gr.KeyUpData):\n", " return {\n", " \"component value\": value,\n", " \"input value\": key_up_data.input_value,\n", " \"key\": key_up_data.key\n", " }\n", "\n", "with gr.Blocks() as demo:\n", " d = gr.Dropdown([\"abc\", \"def\"], allow_custom_value=True)\n", " t = gr.JSON()\n", " d.key_up(test, d, t)\n", "\n", "if __name__ == \"__main__\":\n", " demo.launch()"]}], "metadata": {}, "nbformat": 4, "nbformat_minor": 5}

View File

@ -0,0 +1,16 @@
import gradio as gr
def test(value, key_up_data: gr.KeyUpData):
return {
"component value": value,
"input value": key_up_data.input_value,
"key": key_up_data.key
}
with gr.Blocks() as demo:
d = gr.Dropdown(["abc", "def"], allow_custom_value=True)
t = gr.JSON()
d.key_up(test, d, t)
if __name__ == "__main__":
demo.launch()

View File

@ -59,7 +59,7 @@ from gradio.components import (
from gradio.components.audio import WaveformOptions
from gradio.components.image_editor import Brush, Eraser
from gradio.data_classes import FileData
from gradio.events import EventData, LikeData, SelectData, on
from gradio.events import EventData, KeyUpData, LikeData, SelectData, on
from gradio.exceptions import Error
from gradio.external import load
from gradio.flagging import (

View File

@ -19,7 +19,14 @@ class Dropdown(FormComponent):
Demos: sentence_builder, titanic_survival
"""
EVENTS = [Events.change, Events.input, Events.select, Events.focus, Events.blur]
EVENTS = [
Events.change,
Events.input,
Events.select,
Events.focus,
Events.blur,
Events.key_up,
]
def __init__(
self,

View File

@ -122,6 +122,21 @@ class SelectData(EventData):
"""
class KeyUpData(EventData):
def __init__(self, target: Block | None, data: Any):
super().__init__(target, data)
self.key: str = data["key"]
"""
The key that was pressed.
"""
self.input_value: str = data["input_value"]
"""
The displayed value in the input textbox after the key was pressed. This may be different than the `value`
attribute of the component itself, as the `value` attribute of some components (e.g. Dropdown) are not updated
until the user presses Enter.
"""
@dataclasses.dataclass
class EventListenerMethod:
block: Block | None
@ -516,6 +531,10 @@ class Events:
"load",
doc="This listener is triggered when the {{ component }} initially loads in the browser.",
)
key_up = EventListener(
"key_up",
doc="This listener is triggered when the user presses a key while the {{ component }} is focused.",
)
class LikeData(EventData):

View File

@ -529,7 +529,7 @@
if (
instance_map[id].props.interactive &&
status.stage === "pending" &&
dep.targets[0][1] !== "focus"
!["focus", "key_up"].includes(dep.targets[0][1])
) {
pending_outputs.push(id);
instance_map[id].props.interactive = false;

View File

@ -5,7 +5,7 @@
</script>
<script lang="ts">
import type { Gradio, SelectData } from "@gradio/utils";
import type { Gradio, KeyUpData, SelectData } from "@gradio/utils";
import Multiselect from "./shared/Multiselect.svelte";
import Dropdown from "./shared/Dropdown.svelte";
import { Block } from "@gradio/atoms";
@ -35,6 +35,7 @@
select: SelectData;
blur: never;
focus: never;
key_up: KeyUpData;
}>;
export let interactive: boolean;
</script>
@ -72,6 +73,7 @@
on:select={(e) => gradio.dispatch("select", e.detail)}
on:blur={() => gradio.dispatch("blur")}
on:focus={() => gradio.dispatch("focus")}
on:key_up={() => gradio.dispatch("key_up")}
disabled={!interactive}
/>
{:else}
@ -90,6 +92,7 @@
on:select={(e) => gradio.dispatch("select", e.detail)}
on:blur={() => gradio.dispatch("blur")}
on:focus={() => gradio.dispatch("focus")}
on:key_up={(e) => gradio.dispatch("key_up", e.detail)}
disabled={!interactive}
/>
{/if}

View File

@ -74,8 +74,8 @@ describe("Dropdown", () => {
expect(options[1]).toContainHTML("name2");
});
test("editing the textbox value should filter the options", async () => {
const { getByLabelText, getAllByTestId } = await render(Dropdown, {
test("editing the textbox value should trigger the type event and filter the options", async () => {
const { getByLabelText, listen, getAllByTestId } = await render(Dropdown, {
show_label: true,
loading_status,
max_choices: 10,
@ -89,6 +89,8 @@ describe("Dropdown", () => {
interactive: true
});
const key_up_event = listen("key_up");
const item: HTMLInputElement = getByLabelText(
"Dropdown"
) as HTMLInputElement;
@ -100,10 +102,12 @@ describe("Dropdown", () => {
item.value = "";
await event.keyboard("z");
const options_new = getAllByTestId("dropdown-option");
expect(options_new).toHaveLength(1);
expect(options[0]).toContainHTML("zebra");
await expect(options_new).toHaveLength(1);
await expect(options[0]).toContainHTML("zebra");
await assert.equal(key_up_event.callCount, 1);
});
test("blurring the textbox should cancel the filter", async () => {

View File

@ -3,7 +3,7 @@
import { createEventDispatcher, afterUpdate } from "svelte";
import { BlockTitle } from "@gradio/atoms";
import { DropdownArrow } from "@gradio/icons";
import type { SelectData } from "@gradio/utils";
import type { SelectData, KeyUpData } from "@gradio/utils";
import { handle_filter, handle_change, handle_shared_keys } from "./utils";
export let label: string;
@ -42,6 +42,7 @@
select: SelectData;
blur: undefined;
focus: undefined;
key_up: KeyUpData;
}>();
// Setting the initial value of the dropdown
@ -211,6 +212,11 @@
bind:value={input_text}
bind:this={filter_input}
on:keydown={handle_key_down}
on:keyup={(e) =>
dispatch("key_up", {
key: e.key,
input_value: input_text
})}
on:blur={handle_blur}
on:focus={handle_focus}
readonly={!filterable}

View File

@ -3,7 +3,7 @@
import { _, number } from "svelte-i18n";
import { BlockTitle } from "@gradio/atoms";
import { Remove, DropdownArrow } from "@gradio/icons";
import type { SelectData, I18nFormatter } from "@gradio/utils";
import type { KeyUpData, SelectData, I18nFormatter } from "@gradio/utils";
import DropdownOptions from "./DropdownOptions.svelte";
import { handle_filter, handle_change, handle_shared_keys } from "./utils";
@ -42,6 +42,7 @@
select: SelectData;
blur: undefined;
focus: undefined;
key_up: KeyUpData;
}>();
// Setting the initial value of the multiselect dropdown
@ -259,6 +260,11 @@
bind:value={input_text}
bind:this={filter_input}
on:keydown={handle_key_down}
on:keyup={(e) =>
dispatch("key_up", {
key: e.key,
input_value: input_text
})}
on:blur={handle_blur}
on:focus={handle_focus}
readonly={!filterable}

View File

@ -11,6 +11,11 @@ export interface LikeData {
liked?: boolean;
}
export interface KeyUpData {
key: string;
input_value: string;
}
export interface ShareData {
description: string;
title?: string;