Spaces:
Running
Running
# The Frontend 🌐⭐️ | |
This guide will cover everything you need to know to implement your custom component's frontend. | |
Tip: Gradio components use Svelte. Writing Svelte is fun! If you're not familiar with it, we recommend checking out their interactive [guide](https://learn.svelte.dev/tutorial/welcome-to-svelte). | |
## The directory structure | |
The frontend code should have, at minimum, three files: | |
* `Index.svelte`: This is the main export and where your component's layout and logic should live. | |
* `Example.svelte`: This is where the example view of the component is defined. | |
Feel free to add additional files and subdirectories. | |
If you want to export any additional modules, remember to modify the `package.json` file | |
```json | |
"exports": { | |
".": "./Index.svelte", | |
"./example": "./Example.svelte", | |
"./package.json": "./package.json" | |
}, | |
``` | |
## The Index.svelte file | |
Your component should expose the following props that will be passed down from the parent Gradio application. | |
```typescript | |
import type { LoadingStatus } from "@gradio/statustracker"; | |
import type { Gradio } from "@gradio/utils"; | |
export let gradio: Gradio<{ | |
event_1: never; | |
event_2: never; | |
}>; | |
export let elem_id = ""; | |
export let elem_classes: string[] = []; | |
export let scale: number | null = null; | |
export let min_width: number | undefined = undefined; | |
export let loading_status: LoadingStatus | undefined = undefined; | |
export let mode: "static" | "interactive"; | |
``` | |
* `elem_id` and `elem_classes` allow Gradio app developers to target your component with custom CSS and JavaScript from the Python `Blocks` class. | |
* `scale` and `min_width` allow Gradio app developers to control how much space your component takes up in the UI. | |
* `loading_status` is used to display a loading status over the component when it is the output of an event. | |
* `mode` is how the parent Gradio app tells your component whether the `interactive` or `static` version should be displayed. | |
* `gradio`: The `gradio` object is created by the parent Gradio app. It stores some application-level configuration that will be useful in your component, like internationalization. You must use it to dispatch events from your component. | |
A minimal `Index.svelte` file would look like: | |
```svelte | |
<script lang="ts"> | |
import type { LoadingStatus } from "@gradio/statustracker"; | |
import { Block } from "@gradio/atoms"; | |
import { StatusTracker } from "@gradio/statustracker"; | |
import type { Gradio } from "@gradio/utils"; | |
export let gradio: Gradio<{ | |
event_1: never; | |
event_2: never; | |
}>; | |
export let value = ""; | |
export let elem_id = ""; | |
export let elem_classes: string[] = []; | |
export let scale: number | null = null; | |
export let min_width: number | undefined = undefined; | |
export let loading_status: LoadingStatus | undefined = undefined; | |
export let mode: "static" | "interactive"; | |
</script> | |
<Block | |
visible={true} | |
{elem_id} | |
{elem_classes} | |
{scale} | |
{min_width} | |
allow_overflow={false} | |
padding={true} | |
> | |
{#if loading_status} | |
<StatusTracker | |
autoscroll={gradio.autoscroll} | |
i18n={gradio.i18n} | |
{...loading_status} | |
/> | |
{/if} | |
<p>{value}</p> | |
</Block> | |
``` | |
## The Example.svelte file | |
The `Example.svelte` file should expose the following props: | |
```typescript | |
export let value: string; | |
export let type: "gallery" | "table"; | |
export let selected = false; | |
export let index: number; | |
``` | |
* `value`: The example value that should be displayed. | |
* `type`: This is a variable that can be either `"gallery"` or `"table"` depending on how the examples are displayed. The `"gallery"` form is used when the examples correspond to a single input component, while the `"table"` form is used when a user has multiple input components, and the examples need to populate all of them. | |
* `selected`: You can also adjust how the examples are displayed if a user "selects" a particular example by using the selected variable. | |
* `index`: The current index of the selected value. | |
* Any additional props your "non-example" component takes! | |
This is the `Example.svelte` file for the code `Radio` component: | |
```svelte | |
<script lang="ts"> | |
export let value: string; | |
export let type: "gallery" | "table"; | |
export let selected = false; | |
</script> | |
<div | |
class:table={type === "table"} | |
class:gallery={type === "gallery"} | |
class:selected | |
> | |
{value} | |
</div> | |
<style> | |
.gallery { | |
padding: var(--size-1) var(--size-2); | |
} | |
</style> | |
``` | |
## Handling Files | |
If your component deals with files, these files **should** be uploaded to the backend server. | |
The `@gradio/client` npm package provides the `upload` and `prepare_files` utility functions to help you do this. | |
The `prepare_files` function will convert the browser's `File` datatype to gradio's internal `FileData` type. | |
You should use the `FileData` data in your component to keep track of uploaded files. | |
The `upload` function will upload an array of `FileData` values to the server. | |
Here's an example of loading files from an `<input>` element when its value changes. | |
```svelte | |
<script lang="ts"> | |
import { upload, prepare_files, type FileData } from "@gradio/client"; | |
export let root; | |
export let value; | |
let uploaded_files; | |
async function handle_upload(file_data: FileData[]): Promise<void> { | |
await tick(); | |
uploaded_files = await upload(file_data, root); | |
} | |
async function loadFiles(files: FileList): Promise<void> { | |
let _files: File[] = Array.from(files); | |
if (!files.length) { | |
return; | |
} | |
if (file_count === "single") { | |
_files = [files[0]]; | |
} | |
let file_data = await prepare_files(_files); | |
await handle_upload(file_data); | |
} | |
async function loadFilesFromUpload(e: Event): Promise<void> { | |
const target = e.target; | |
if (!target.files) return; | |
await loadFiles(target.files); | |
} | |
</script> | |
<input | |
type="file" | |
on:change={loadFilesFromUpload} | |
multiple={true} | |
/> | |
``` | |
The component exposes a prop named `root`. | |
This is passed down by the parent gradio app and it represents the base url that the files will be uploaded to and fetched from. | |
For WASM support, you should get the upload function from the `Context` and pass that as the third parameter of the `upload` function. | |
```typescript | |
<script lang="ts"> | |
import { getContext } from "svelte"; | |
const upload_fn = getContext<typeof upload_files>("upload_files"); | |
async function handle_upload(file_data: FileData[]): Promise<void> { | |
await tick(); | |
await upload(file_data, root, upload_fn); | |
} | |
</script> | |
``` | |
## Leveraging Existing Gradio Components | |
Most of Gradio's frontend components are published on [npm](https://www.npmjs.com/), the javascript package repository. | |
This means that you can use them to save yourself time while incorporating common patterns in your component, like uploading files. | |
For example, the `@gradio/upload` package has `Upload` and `ModifyUpload` components for properly uploading files to the Gradio server. | |
Here is how you can use them to create a user interface to upload and display PDF files. | |
```svelte | |
<script> | |
import { type FileData, Upload, ModifyUpload } from "@gradio/upload"; | |
import { Empty, UploadText, BlockLabel } from "@gradio/atoms"; | |
</script> | |
<BlockLabel Icon={File} label={label || "PDF"} /> | |
{#if value === null && interactive} | |
<Upload | |
filetype="application/pdf" | |
on:load={handle_load} | |
{root} | |
> | |
<UploadText type="file" i18n={gradio.i18n} /> | |
</Upload> | |
{:else if value !== null} | |
{#if interactive} | |
<ModifyUpload i18n={gradio.i18n} on:clear={handle_clear}/> | |
{/if} | |
<iframe title={value.orig_name || "PDF"} src={value.data} height="{height}px" width="100%"></iframe> | |
{:else} | |
<Empty size="large"> <File/> </Empty> | |
{/if} | |
``` | |
You can also combine existing Gradio components to create entirely unique experiences. | |
Like rendering a gallery of chatbot conversations. | |
The possibilities are endless, please read the documentation on our javascript packages [here](https://gradio.app/main/docs/js). | |
We'll be adding more packages and documentation over the coming weeks! | |
## Matching Gradio Core's Design System | |
You can explore our component library via Storybook. You'll be able to interact with our components and see them in their various states. | |
For those interested in design customization, we provide the CSS variables consisting of our color palette, radii, spacing, and the icons we use - so you can easily match up your custom component with the style of our core components. This Storybook will be regularly updated with any new additions or changes. | |
[Storybook Link](https://gradio.app/main/docs/js/storybook) | |
## Custom configuration | |
If you want to make use of the vast vite ecosystem, you can use the `gradio.config.js` file to configure your component's build process. This allows you to make use of tools like tailwindcss, mdsvex, and more. | |
Currently, it is possible to configure the following: | |
Vite options: | |
- `plugins`: A list of vite plugins to use. | |
Svelte options: | |
- `preprocess`: A list of svelte preprocessors to use. | |
- `extensions`: A list of file extensions to compile to `.svelte` files. | |
- `build.target`: The target to build for, this may be necessary to support newer javascript features. See the [esbuild docs](https://esbuild.github.io/api/#target) for more information. | |
The `gradio.config.js` file should be placed in the root of your component's `frontend` directory. A default config file is created for you when you create a new component. But you can also create your own config file, if one doesn't exist, and use it to customize your component's build process. | |
### Example for a Vite plugin | |
Custom components can use Vite plugins to customize the build process. Check out the [Vite Docs](https://vitejs.dev/guide/using-plugins.html) for more information. | |
Here we configure [TailwindCSS](https://tailwindcss.com), a utility-first CSS framework. Setup is easiest using the version 4 prerelease. | |
``` | |
npm install tailwindcss@next @tailwindcss/vite@next | |
``` | |
In `gradio.config.js`: | |
```typescript | |
import tailwindcss from "@tailwindcss/vite"; | |
export default { | |
plugins: [tailwindcss()] | |
}; | |
``` | |
Then create a `style.css` file with the following content: | |
```css | |
@import "tailwindcss"; | |
``` | |
Import this file into `Index.svelte`. Note, that you need to import the css file containing `@import` and cannot just use a `<style>` tag and use `@import` there. | |
```svelte | |
<script lang="ts"> | |
[...] | |
import "./style.css"; | |
[...] | |
</script> | |
``` | |
### Example for Svelte options | |
In `gradio.config.js` you can also specify a some Svelte options to apply to the Svelte compilation. In this example we will add support for [`mdsvex`](https://mdsvex.pngwn.io), a Markdown preprocessor for Svelte. | |
In order to do this we will need to add a [Svelte Preprocessor](https://svelte.dev/docs/svelte-compiler#preprocess) to the `svelte` object in `gradio.config.js` and configure the [`extensions`](https://github.com/sveltejs/vite-plugin-svelte/blob/HEAD/docs/config.md#config-file) field. Other options are not currently supported. | |
First, install the `mdsvex` plugin: | |
```bash | |
npm install mdsvex | |
``` | |
Then add the following to `gradio.config.js`: | |
```typescript | |
import { mdsvex } from "mdsvex"; | |
export default { | |
svelte: { | |
preprocess: [ | |
mdsvex() | |
], | |
extensions: [".svelte", ".svx"] | |
} | |
}; | |
``` | |
Now we can create `mdsvex` documents in our component's `frontend` directory and they will be compiled to `.svelte` files. | |
```md | |
<!-- HelloWorld.svx --> | |
<script lang="ts"> | |
import { Block } from "@gradio/atoms"; | |
export let title = "Hello World"; | |
</script> | |
<Block label="Hello World"> | |
# {title} | |
This is a markdown file. | |
</Block> | |
``` | |
We can then use the `HelloWorld.svx` file in our components: | |
```svelte | |
<script lang="ts"> | |
import HelloWorld from "./HelloWorld.svx"; | |
</script> | |
<HelloWorld /> | |
``` | |
## Conclusion | |
You now how to create delightful frontends for your components! | |