mirror of
https://github.com/gradio-app/gradio.git
synced 2025-03-31 12:20:26 +08:00
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:
parent
65437ce832
commit
065c5b163c
8
.changeset/metal-moons-sleep.md
Normal file
8
.changeset/metal-moons-sleep.md
Normal 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()`
|
1
demo/dropdown_key_up/run.ipynb
Normal file
1
demo/dropdown_key_up/run.ipynb
Normal 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}
|
16
demo/dropdown_key_up/run.py
Normal file
16
demo/dropdown_key_up/run.py
Normal 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()
|
@ -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 (
|
||||
|
@ -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,
|
||||
|
@ -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):
|
||||
|
@ -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;
|
||||
|
@ -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}
|
||||
|
@ -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 () => {
|
||||
|
@ -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}
|
||||
|
@ -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}
|
||||
|
@ -11,6 +11,11 @@ export interface LikeData {
|
||||
liked?: boolean;
|
||||
}
|
||||
|
||||
export interface KeyUpData {
|
||||
key: string;
|
||||
input_value: string;
|
||||
}
|
||||
|
||||
export interface ShareData {
|
||||
description: string;
|
||||
title?: string;
|
||||
|
Loading…
x
Reference in New Issue
Block a user