mirror of
https://github.com/gradio-app/gradio.git
synced 2025-03-31 12:20:26 +08:00
Add copy button to gr.Markdown
(#8851)
* add copy button * add changeset * add changeset * lint * value tweak --------- Co-authored-by: gradio-pr-bot <gradio-pr-bot@users.noreply.github.com> Co-authored-by: Abubakar Abid <abubakar@huggingface.co>
This commit is contained in:
parent
5622331da7
commit
914b1935de
6
.changeset/kind-deer-create.md
Normal file
6
.changeset/kind-deer-create.md
Normal file
@ -0,0 +1,6 @@
|
||||
---
|
||||
"@gradio/markdown": minor
|
||||
"gradio": minor
|
||||
---
|
||||
|
||||
feat:Add copy button to `gr.Markdown`
|
@ -45,6 +45,7 @@ class Markdown(Component):
|
||||
line_breaks: bool = False,
|
||||
header_links: bool = False,
|
||||
height: int | str | None = None,
|
||||
show_copy_button: bool = False,
|
||||
):
|
||||
"""
|
||||
Parameters:
|
||||
@ -64,6 +65,7 @@ class Markdown(Component):
|
||||
line_breaks: If True, will enable Github-flavored Markdown line breaks in chatbot messages. If False (default), single new lines will be ignored.
|
||||
header_links: If True, will automatically create anchors for headings, displaying a link icon on hover.
|
||||
height: An optional maximum height of this component, specified in pixels if a number is passed, or in CSS units (e.g., '200px') if a stirng is passed in. If context exceeds this height, a scrollbar is added.
|
||||
show_copy_button: If True, includes a copy button to copy the text in the Markdown component. Default is False.
|
||||
"""
|
||||
self.rtl = rtl
|
||||
if latex_delimiters is None:
|
||||
@ -73,6 +75,7 @@ class Markdown(Component):
|
||||
self.line_breaks = line_breaks
|
||||
self.header_links = header_links
|
||||
self.height = height
|
||||
self.show_copy_button = show_copy_button
|
||||
|
||||
super().__init__(
|
||||
label=label,
|
||||
|
@ -32,6 +32,7 @@
|
||||
}[];
|
||||
export let header_links = false;
|
||||
export let height: number | string | undefined = undefined;
|
||||
export let show_copy_button = false;
|
||||
|
||||
$: label, gradio.dispatch("change");
|
||||
</script>
|
||||
@ -63,6 +64,7 @@
|
||||
{line_breaks}
|
||||
{header_links}
|
||||
{height}
|
||||
{show_copy_button}
|
||||
/>
|
||||
</div>
|
||||
</Block>
|
||||
|
@ -84,3 +84,12 @@ in two separate lines.`
|
||||
height: "200px"
|
||||
}}
|
||||
/>
|
||||
|
||||
<Story
|
||||
name="Markdown with Copy Button"
|
||||
args={{
|
||||
value:
|
||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed non risus. Suspendisse lectus tortor, dignissim sit amet, adipiscing nec, ultricies sed, dolor. Cras elementum ultrices diam. Maecenas ligula massa, varius a, semper congue, euismod non, mi. Proin porttitor, orci nec nonummy molestie, enim est eleifend mi, non fermentum diam nisl sit amet erat.",
|
||||
show_copy_button: true
|
||||
}}
|
||||
/>
|
||||
|
@ -15,6 +15,7 @@
|
||||
},
|
||||
"dependencies": {
|
||||
"@gradio/atoms": "workspace:^",
|
||||
"@gradio/icons": "workspace:^",
|
||||
"@gradio/statustracker": "workspace:^",
|
||||
"@gradio/utils": "workspace:^",
|
||||
"@types/dompurify": "^3.0.2",
|
||||
|
@ -1,8 +1,10 @@
|
||||
<script lang="ts">
|
||||
import { createEventDispatcher } from "svelte";
|
||||
import { copy } from "@gradio/utils";
|
||||
import { Copy, Check } from "@gradio/icons";
|
||||
|
||||
import MarkdownCode from "./MarkdownCode.svelte";
|
||||
import { fade } from "svelte/transition";
|
||||
|
||||
export let elem_classes: string[] = [];
|
||||
export let visible = true;
|
||||
@ -18,6 +20,10 @@
|
||||
}[];
|
||||
export let header_links = false;
|
||||
export let height: number | string | undefined = undefined;
|
||||
export let show_copy_button = false;
|
||||
|
||||
let copied = false;
|
||||
let timer: NodeJS.Timeout;
|
||||
|
||||
const dispatch = createEventDispatcher<{ change: undefined }>();
|
||||
|
||||
@ -28,6 +34,21 @@
|
||||
};
|
||||
|
||||
$: value, dispatch("change");
|
||||
|
||||
async function handle_copy(): Promise<void> {
|
||||
if ("clipboard" in navigator) {
|
||||
await navigator.clipboard.writeText(value);
|
||||
copy_feedback();
|
||||
}
|
||||
}
|
||||
|
||||
function copy_feedback(): void {
|
||||
copied = true;
|
||||
if (timer) clearTimeout(timer);
|
||||
timer = setTimeout(() => {
|
||||
copied = false;
|
||||
}, 1000);
|
||||
}
|
||||
</script>
|
||||
|
||||
<div
|
||||
@ -39,6 +60,21 @@
|
||||
use:copy
|
||||
style={height ? `max-height: ${css_units(height)}; overflow-y: auto;` : ""}
|
||||
>
|
||||
{#if show_copy_button}
|
||||
{#if copied}
|
||||
<button
|
||||
in:fade={{ duration: 300 }}
|
||||
aria-label="Copied"
|
||||
aria-roledescription="Text copied"><Check /></button
|
||||
>
|
||||
{:else}
|
||||
<button
|
||||
on:click={handle_copy}
|
||||
aria-label="Copy"
|
||||
aria-roledescription="Copy text"><Copy /></button
|
||||
>
|
||||
{/if}
|
||||
{/if}
|
||||
<MarkdownCode
|
||||
message={value}
|
||||
{latex_delimiters}
|
||||
@ -73,4 +109,25 @@
|
||||
.hide {
|
||||
display: none;
|
||||
}
|
||||
|
||||
button {
|
||||
display: flex;
|
||||
position: absolute;
|
||||
top: -10px;
|
||||
right: -10px;
|
||||
align-items: center;
|
||||
box-shadow: var(--shadow-drop);
|
||||
border: 1px solid var(--color-border-primary);
|
||||
border-top: none;
|
||||
border-right: none;
|
||||
border-radius: var(--block-label-right-radius);
|
||||
background: var(--block-label-background-fill);
|
||||
padding: 5px;
|
||||
width: 22px;
|
||||
height: 22px;
|
||||
overflow: hidden;
|
||||
color: var(--block-label-color);
|
||||
font: var(--font-sans);
|
||||
font-size: var(--button-small-text-size);
|
||||
}
|
||||
</style>
|
||||
|
3
pnpm-lock.yaml
generated
3
pnpm-lock.yaml
generated
@ -1434,6 +1434,9 @@ importers:
|
||||
'@gradio/atoms':
|
||||
specifier: workspace:^
|
||||
version: link:../atoms
|
||||
'@gradio/icons':
|
||||
specifier: workspace:^
|
||||
version: link:../icons
|
||||
'@gradio/statustracker':
|
||||
specifier: workspace:^
|
||||
version: link:../statustracker
|
||||
|
@ -5,6 +5,7 @@ class TestMarkdown:
|
||||
def test_component_functions(self):
|
||||
markdown_component = gr.Markdown("# Let's learn about $x$", label="Markdown")
|
||||
assert markdown_component.get_config()["value"] == "# Let's learn about $x$"
|
||||
assert not markdown_component.get_config()["show_copy_button"]
|
||||
|
||||
def test_in_interface(self):
|
||||
"""
|
||||
@ -14,3 +15,9 @@ class TestMarkdown:
|
||||
input_data = " Here's an [image](https://gradio.app/images/gradio_logo.png)"
|
||||
output_data = iface(input_data)
|
||||
assert output_data == input_data.strip()
|
||||
|
||||
def test_show_copy_button(self):
|
||||
markdown_component = gr.Markdown(
|
||||
"# Let's learn about $x$", show_copy_button=True
|
||||
)
|
||||
assert markdown_component.get_config()["show_copy_button"]
|
||||
|
Loading…
x
Reference in New Issue
Block a user