freddyaboulton HF staff commited on
Commit
5db0478
·
verified ·
1 Parent(s): 0cbeb30

Upload folder using huggingface_hub

Browse files
Dockerfile CHANGED
@@ -5,8 +5,12 @@ WORKDIR /code
5
 
6
  COPY --link --chown=1000 . .
7
 
 
 
 
 
8
  RUN pip install --no-cache-dir -r requirements.txt
9
 
10
  ENV PYTHONUNBUFFERED=1 GRADIO_ALLOW_FLAGGING=never GRADIO_NUM_PORTS=1 GRADIO_SERVER_NAME=0.0.0.0 GRADIO_SERVER_PORT=7860 SYSTEM=spaces
11
 
12
- CMD ["python", "app.py"]
 
5
 
6
  COPY --link --chown=1000 . .
7
 
8
+ RUN mkdir -p /tmp/cache/
9
+ RUN chmod a+rwx -R /tmp/cache/
10
+ ENV TRANSFORMERS_CACHE=/tmp/cache/
11
+
12
  RUN pip install --no-cache-dir -r requirements.txt
13
 
14
  ENV PYTHONUNBUFFERED=1 GRADIO_ALLOW_FLAGGING=never GRADIO_NUM_PORTS=1 GRADIO_SERVER_NAME=0.0.0.0 GRADIO_SERVER_PORT=7860 SYSTEM=spaces
15
 
16
+ CMD ["python", "space.py"]
README.md CHANGED
@@ -1,10 +1,17 @@
1
 
2
  ---
3
- tags: [gradio-custom-component, gradio-template-Chatbot]
4
  title: gradio_multimodalchatbot V0.0.1
5
- colorFrom: purple
6
- colorTo: indigo
7
  sdk: docker
8
  pinned: false
9
  license: apache-2.0
10
  ---
 
 
 
 
 
 
 
 
1
 
2
  ---
3
+ tags: [gradio-custom-component,Chatbot,Multimodal,Media,gradio-template-Chatbot]
4
  title: gradio_multimodalchatbot V0.0.1
5
+ colorFrom: green
6
+ colorTo: blue
7
  sdk: docker
8
  pinned: false
9
  license: apache-2.0
10
  ---
11
+
12
+
13
+ # Name: gradio_multimodalchatbot
14
+
15
+ Description: Display text and media files (audio, video, images) in the same chat message bubble.
16
+
17
+ Install with: pip install gradio_multimodalchatbot
__pycache__/app.cpython-310.pyc CHANGED
Binary files a/__pycache__/app.cpython-310.pyc and b/__pycache__/app.cpython-310.pyc differ
 
app.py CHANGED
@@ -23,7 +23,8 @@ bot_msg3 = {"text": "Here is a video clip of the world",
23
  conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
24
 
25
  with gr.Blocks() as demo:
26
- MultimodalChatbot(value=conversation)
27
 
28
 
29
- demo.launch()
 
 
23
  conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
24
 
25
  with gr.Blocks() as demo:
26
+ MultimodalChatbot(value=conversation, height=800)
27
 
28
 
29
+ if __name__ == "__main__":
30
+ demo.launch()
css.css ADDED
@@ -0,0 +1,157 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ html {
2
+ font-family: Inter;
3
+ font-size: 16px;
4
+ font-weight: 400;
5
+ line-height: 1.5;
6
+ -webkit-text-size-adjust: 100%;
7
+ background: #fff;
8
+ color: #323232;
9
+ -webkit-font-smoothing: antialiased;
10
+ -moz-osx-font-smoothing: grayscale;
11
+ text-rendering: optimizeLegibility;
12
+ }
13
+
14
+ :root {
15
+ --space: 1;
16
+ --vspace: calc(var(--space) * 1rem);
17
+ --vspace-0: calc(3 * var(--space) * 1rem);
18
+ --vspace-1: calc(2 * var(--space) * 1rem);
19
+ --vspace-2: calc(1.5 * var(--space) * 1rem);
20
+ --vspace-3: calc(0.5 * var(--space) * 1rem);
21
+ }
22
+
23
+ .app {
24
+ max-width: 748px !important;
25
+ }
26
+
27
+ .prose p {
28
+ margin: var(--vspace) 0;
29
+ line-height: var(--vspace * 2);
30
+ font-size: 1rem;
31
+ }
32
+
33
+ code {
34
+ font-family: "Inconsolata", sans-serif;
35
+ font-size: 16px;
36
+ }
37
+
38
+ h1,
39
+ h1 code {
40
+ font-weight: 400;
41
+ line-height: calc(2.5 / var(--space) * var(--vspace));
42
+ }
43
+
44
+ h1 code {
45
+ background: none;
46
+ border: none;
47
+ letter-spacing: 0.05em;
48
+ padding-bottom: 5px;
49
+ position: relative;
50
+ padding: 0;
51
+ }
52
+
53
+ h2 {
54
+ margin: var(--vspace-1) 0 var(--vspace-2) 0;
55
+ line-height: 1em;
56
+ }
57
+
58
+ h3,
59
+ h3 code {
60
+ margin: var(--vspace-1) 0 var(--vspace-2) 0;
61
+ line-height: 1em;
62
+ }
63
+
64
+ h4,
65
+ h5,
66
+ h6 {
67
+ margin: var(--vspace-3) 0 var(--vspace-3) 0;
68
+ line-height: var(--vspace);
69
+ }
70
+
71
+ .bigtitle,
72
+ h1,
73
+ h1 code {
74
+ font-size: calc(8px * 4.5);
75
+ word-break: break-word;
76
+ }
77
+
78
+ .title,
79
+ h2,
80
+ h2 code {
81
+ font-size: calc(8px * 3.375);
82
+ font-weight: lighter;
83
+ word-break: break-word;
84
+ border: none;
85
+ background: none;
86
+ }
87
+
88
+ .subheading1,
89
+ h3,
90
+ h3 code {
91
+ font-size: calc(8px * 1.8);
92
+ font-weight: 600;
93
+ border: none;
94
+ background: none;
95
+ letter-spacing: 0.1em;
96
+ text-transform: uppercase;
97
+ }
98
+
99
+ h2 code {
100
+ padding: 0;
101
+ position: relative;
102
+ letter-spacing: 0.05em;
103
+ }
104
+
105
+ blockquote {
106
+ font-size: calc(8px * 1.1667);
107
+ font-style: italic;
108
+ line-height: calc(1.1667 * var(--vspace));
109
+ margin: var(--vspace-2) var(--vspace-2);
110
+ }
111
+
112
+ .subheading2,
113
+ h4 {
114
+ font-size: calc(8px * 1.4292);
115
+ text-transform: uppercase;
116
+ font-weight: 600;
117
+ }
118
+
119
+ .subheading3,
120
+ h5 {
121
+ font-size: calc(8px * 1.2917);
122
+ line-height: calc(1.2917 * var(--vspace));
123
+
124
+ font-weight: lighter;
125
+ text-transform: uppercase;
126
+ letter-spacing: 0.15em;
127
+ }
128
+
129
+ h6 {
130
+ font-size: calc(8px * 1.1667);
131
+ font-size: 1.1667em;
132
+ font-weight: normal;
133
+ font-style: italic;
134
+ font-family: "le-monde-livre-classic-byol", serif !important;
135
+ letter-spacing: 0px !important;
136
+ }
137
+
138
+ #start .md > *:first-child {
139
+ margin-top: 0;
140
+ }
141
+
142
+ h2 + h3 {
143
+ margin-top: 0;
144
+ }
145
+
146
+ .md hr {
147
+ border: none;
148
+ border-top: 1px solid var(--block-border-color);
149
+ margin: var(--vspace-2) 0 var(--vspace-2) 0;
150
+ }
151
+ .prose ul {
152
+ margin: var(--vspace-2) 0 var(--vspace-1) 0;
153
+ }
154
+
155
+ .gap {
156
+ gap: 0;
157
+ }
requirements.txt CHANGED
@@ -1 +1 @@
1
- gradio_multimodalchatbot-0.0.1-py3-none-any.whl
 
1
+ gradio_multimodalchatbot==0.0.1
space.py ADDED
@@ -0,0 +1,171 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+
2
+ import gradio as gr
3
+ from app import demo as app
4
+ import os
5
+
6
+ _docs = {'MultimodalChatbot': {'description': 'Displays a chatbot output showing both user submitted messages and responses. Supports a subset of Markdown including bold, italics, code, tables. Also supports audio/video/image files, which are displayed in the MultimodalChatbot, and other kinds of files which are displayed as links.\n', 'members': {'__init__': {'value': {'type': 'list[\n list[\n str\n | tuple[str]\n | tuple[str | pathlib.Path, str]\n | None\n ]\n ]\n | Callable\n | None', 'default': 'None', 'description': 'Default value to show in chatbot. If callable, the function will be called whenever the app loads to set the initial value of the component.'}, 'label': {'type': 'str | None', 'default': 'None', 'description': 'The label for this component. Appears above the component and is also used as the header if there are a table of examples for this component. If None and used in a `gr.Interface`, the label will be the name of the parameter this component is assigned to.'}, 'every': {'type': 'float | None', 'default': 'None', 'description': "If `value` is a callable, run the function 'every' number of seconds while the client connection is open. Has no effect otherwise. Queue must be enabled. The event can be accessed (e.g. to cancel it) via this component's .load_event attribute."}, 'show_label': {'type': 'bool | None', 'default': 'None', 'description': 'if True, will display label.'}, 'container': {'type': 'bool', 'default': 'True', 'description': 'If True, will place the component in a container - providing some extra padding around the border.'}, 'scale': {'type': 'int | None', 'default': 'None', 'description': 'relative width compared to adjacent Components in a Row. For example, if Component A has scale=2, and Component B has scale=1, A will be twice as wide as B. Should be an integer.'}, 'min_width': {'type': 'int', 'default': '160', 'description': 'minimum pixel width, will wrap if not sufficient screen space to satisfy this value. If a certain scale value results in this Component being narrower than min_width, the min_width parameter will be respected first.'}, 'visible': {'type': 'bool', 'default': 'True', 'description': 'If False, component will be hidden.'}, 'elem_id': {'type': 'str | None', 'default': 'None', 'description': 'An optional string that is assigned as the id of this component in the HTML DOM. Can be used for targeting CSS styles.'}, 'elem_classes': {'type': 'list[str] | str | None', 'default': 'None', 'description': 'An optional list of strings that are assigned as the classes of this component in the HTML DOM. Can be used for targeting CSS styles.'}, 'render': {'type': 'bool', 'default': 'True', 'description': 'If False, component will not render be rendered in the Blocks context. Should be used if the intention is to assign event listeners now but render the component later.'}, 'height': {'type': 'int | None', 'default': 'None', 'description': 'height of the component in pixels.'}, 'latex_delimiters': {'type': 'list[dict[str, str | bool]] | None', 'default': 'None', 'description': 'A list of dicts of the form {"left": open delimiter (str), "right": close delimiter (str), "display": whether to display in newline (bool)} that will be used to render LaTeX expressions. If not provided, `latex_delimiters` is set to `[{ "left": "$$", "right": "$$", "display": True }]`, so only expressions enclosed in $$ delimiters will be rendered as LaTeX, and in a new line. Pass in an empty list to disable LaTeX rendering. For more information, see the [KaTeX documentation](https://katex.org/docs/autorender.html).'}, 'rtl': {'type': 'bool', 'default': 'False', 'description': 'If True, sets the direction of the rendered text to right-to-left. Default is False, which renders text left-to-right.'}, 'show_share_button': {'type': 'bool | None', 'default': 'None', 'description': 'If True, will show a share icon in the corner of the component that allows user to share outputs to Hugging Face Spaces Discussions. If False, icon does not appear. If set to None (default behavior), then the icon appears if this Gradio app is launched on Spaces, but not otherwise.'}, 'show_copy_button': {'type': 'bool', 'default': 'False', 'description': 'If True, will show a copy button for each chatbot message.'}, 'avatar_images': {'type': 'tuple[\n str | pathlib.Path | None, str | pathlib.Path | None\n ]\n | None', 'default': 'None', 'description': 'Tuple of two avatar image paths or URLs for user and bot (in that order). Pass None for either the user or bot image to skip. Must be within the working directory of the Gradio app or an external URL.'}, 'sanitize_html': {'type': 'bool', 'default': 'True', 'description': 'If False, will disable HTML sanitization for chatbot messages. This is not recommended, as it can lead to security vulnerabilities.'}, 'render_markdown': {'type': 'bool', 'default': 'True', 'description': 'If False, will disable Markdown rendering for chatbot messages.'}, 'bubble_full_width': {'type': 'bool', 'default': 'True', 'description': 'If False, the chat bubble will fit to the content of the message. If True (default), the chat bubble will be the full width of the component.'}, 'line_breaks': {'type': 'bool', 'default': 'True', 'description': 'If True (default), will enable Github-flavored Markdown line breaks in chatbot messages. If False, single new lines will be ignored. Only applies if `render_markdown` is True.'}, 'likeable': {'type': 'bool', 'default': 'False', 'description': 'Whether the chat messages display a like or dislike button. Set automatically by the .like method but has to be present in the signature for it to show up in the config.'}, 'layout': {'type': '"panel" | "bubble" | None', 'default': 'None', 'description': 'If "panel", will display the chatbot in a llm style layout. If "bubble", will display the chatbot with message bubbles, with the user and bot messages on alterating sides. Will default to "bubble".'}}, 'postprocess': {'value': {'type': 'list[\n list[str | tuple[str] | tuple[str, str] | None] | tuple\n]', 'description': None}}, 'preprocess': {'return': {'type': 'list[MultimodalMessage] | None', 'description': None}, 'value': None}}, 'events': {'change': {'type': None, 'default': None, 'description': 'Triggered when the value of the MultimodalChatbot changes either because of user input (e.g. a user types in a textbox) OR because of a function update (e.g. an image receives a value from the output of an event trigger). See `.input()` for a listener that is only triggered by user input.'}, 'select': {'type': None, 'default': None, 'description': 'Event listener for when the user selects or deselects the MultimodalChatbot. Uses event data gradio.SelectData to carry `value` referring to the label of the MultimodalChatbot, and `selected` to refer to state of the MultimodalChatbot. See EventData documentation on how to use this event data'}, 'like': {'type': None, 'default': None, 'description': 'This listener is triggered when the user likes/dislikes from within the MultimodalChatbot. This event has EventData of type gradio.LikeData that carries information, accessible through LikeData.index and LikeData.value. See EventData documentation on how to use this event data.'}}}, '__meta__': {'additional_interfaces': {'MultimodalMessage': {'source': 'class MultimodalMessage(GradioModel):\n text: Optional[str] = None\n files: Optional[List[FileMessage]] = None', 'refs': ['FileMessage']}, 'FileMessage': {'source': 'class FileMessage(GradioModel):\n file: FileData\n alt_text: Optional[str] = None'}}, 'user_fn_refs': {'MultimodalChatbot': ['MultimodalMessage']}}}
7
+
8
+ abs_path = os.path.join(os.path.dirname(__file__), "css.css")
9
+
10
+ with gr.Blocks(
11
+ css=abs_path,
12
+ theme=gr.themes.Default(
13
+ font_mono=[
14
+ gr.themes.GoogleFont("Inconsolata"),
15
+ "monospace",
16
+ ],
17
+ ),
18
+ ) as demo:
19
+ gr.Markdown(
20
+ """
21
+ # `gradio_multimodalchatbot`
22
+
23
+ <div style="display: flex; gap: 7px;">
24
+ <a href="https://pypi.org/project/gradio_multimodalchatbot/" target="_blank"><img alt="PyPI - Version" src="https://img.shields.io/pypi/v/gradio_multimodalchatbot"></a> <a href="https://huggingface.co/spaces/freddyaboulton/gradio_multimodalchatbot//discussions" target="_blank"><img alt="Static Badge" src="https://img.shields.io/badge/%F0%9F%A4%97%20Discuss-%23097EFF?style=flat&logoColor=black"></a>
25
+ </div>
26
+
27
+ Display text and media files (audio, video, images) in the same chat message bubble.
28
+ """, elem_classes=["md-custom"], header_links=True)
29
+ app.render()
30
+ gr.Markdown(
31
+ """
32
+ ## Installation
33
+
34
+ ```bash
35
+ pip install gradio_multimodalchatbot
36
+ ```
37
+
38
+ ## Usage
39
+
40
+ ```python
41
+
42
+ import gradio as gr
43
+ from gradio_multimodalchatbot import MultimodalChatbot
44
+ from gradio.data_classes import FileData
45
+
46
+ user_msg1 = {"text": "Hello, what is in this image?",
47
+ "files": [{"file": FileData(path="https://gradio-builds.s3.amazonaws.com/diffusion_image/cute_dog.jpg")}]
48
+ }
49
+ bot_msg1 = {"text": "It is a very cute dog",
50
+ "files": []}
51
+
52
+ user_msg2 = {"text": "Describe this audio clip please.",
53
+ "files": [{"file": FileData(path="cantina.wav")}]}
54
+ bot_msg2 = {"text": "It is the cantina song from Star Wars",
55
+ "files": []}
56
+
57
+ user_msg3 = {"text": "Give me a video clip please.",
58
+ "files": []}
59
+ bot_msg3 = {"text": "Here is a video clip of the world",
60
+ "files": [{"file": FileData(path="world.mp4")},
61
+ {"file": FileData(path="cantina.wav")}]}
62
+
63
+ conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
64
+
65
+ with gr.Blocks() as demo:
66
+ MultimodalChatbot(value=conversation, height=800)
67
+
68
+
69
+ if __name__ == "__main__":
70
+ demo.launch()
71
+
72
+ ```
73
+ """, elem_classes=["md-custom"], header_links=True)
74
+
75
+
76
+ gr.Markdown("""
77
+ ## `MultimodalChatbot`
78
+
79
+ ### Initialization
80
+ """, elem_classes=["md-custom"], header_links=True)
81
+
82
+ gr.ParamViewer(value=_docs["MultimodalChatbot"]["members"]["__init__"], linkify=['MultimodalMessage', 'FileMessage'])
83
+
84
+
85
+ gr.Markdown("### Events")
86
+ gr.ParamViewer(value=_docs["MultimodalChatbot"]["events"], linkify=['Event'])
87
+
88
+
89
+
90
+
91
+ gr.Markdown("""
92
+
93
+ ### User function
94
+
95
+ The impact on the users predict function varies depending on whether the component is used as an input or output for an event (or both).
96
+
97
+ - When used as an Input, the component only impacts the input signature of the user function.
98
+ - When used as an output, the component only impacts the return signature of the user function.
99
+
100
+ The code snippet below is accurate in cases where the component is used as both an input and an output.
101
+
102
+
103
+
104
+ ```python
105
+ def predict(
106
+ value: list[MultimodalMessage] | None
107
+ ) -> list[
108
+ list[str | tuple[str] | tuple[str, str] | None] | tuple
109
+ ]:
110
+ return value
111
+ ```
112
+ """, elem_classes=["md-custom", "MultimodalChatbot-user-fn"], header_links=True)
113
+
114
+
115
+
116
+
117
+ code_MultimodalMessage = gr.Markdown("""
118
+ ## `MultimodalMessage`
119
+ ```python
120
+ class MultimodalMessage(GradioModel):
121
+ text: Optional[str] = None
122
+ files: Optional[List[FileMessage]] = None
123
+ ```""", elem_classes=["md-custom", "MultimodalMessage"], header_links=True)
124
+
125
+ code_FileMessage = gr.Markdown("""
126
+ ## `FileMessage`
127
+ ```python
128
+ class FileMessage(GradioModel):
129
+ file: FileData
130
+ alt_text: Optional[str] = None
131
+ ```""", elem_classes=["md-custom", "FileMessage"], header_links=True)
132
+
133
+ demo.load(None, js=r"""function() {
134
+ const refs = {
135
+ MultimodalMessage: ['FileMessage'],
136
+ FileMessage: [], };
137
+ const user_fn_refs = {
138
+ MultimodalChatbot: ['MultimodalMessage'], };
139
+ requestAnimationFrame(() => {
140
+
141
+ Object.entries(user_fn_refs).forEach(([key, refs]) => {
142
+ if (refs.length > 0) {
143
+ const el = document.querySelector(`.${key}-user-fn`);
144
+ if (!el) return;
145
+ refs.forEach(ref => {
146
+ el.innerHTML = el.innerHTML.replace(
147
+ new RegExp("\\b"+ref+"\\b", "g"),
148
+ `<a href="#h-${ref.toLowerCase()}">${ref}</a>`
149
+ );
150
+ })
151
+ }
152
+ })
153
+
154
+ Object.entries(refs).forEach(([key, refs]) => {
155
+ if (refs.length > 0) {
156
+ const el = document.querySelector(`.${key}`);
157
+ if (!el) return;
158
+ refs.forEach(ref => {
159
+ el.innerHTML = el.innerHTML.replace(
160
+ new RegExp("\\b"+ref+"\\b", "g"),
161
+ `<a href="#h-${ref.toLowerCase()}">${ref}</a>`
162
+ );
163
+ })
164
+ }
165
+ })
166
+ })
167
+ }
168
+
169
+ """)
170
+
171
+ demo.launch()
src/README.md CHANGED
@@ -1,10 +1,419 @@
1
 
2
- # gradio_multimodalchatbot
3
- A Custom Gradio component.
4
 
5
- ## Example usage
 
 
 
 
 
 
 
 
6
 
7
  ```python
 
8
  import gradio as gr
9
  from gradio_multimodalchatbot import MultimodalChatbot
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
10
  ```
 
1
 
2
+ # `gradio_multimodalchatbot`
3
+ <a href="https://pypi.org/project/gradio_multimodalchatbot/" target="_blank"><img alt="PyPI - Version" src="https://img.shields.io/pypi/v/gradio_multimodalchatbot"></a> <a href="https://huggingface.co/spaces/freddyaboulton/gradio_multimodalchatbot//discussions" target="_blank"><img alt="Static Badge" src="https://img.shields.io/badge/%F0%9F%A4%97%20Discuss-%23097EFF?style=flat&logoColor=black"></a>
4
 
5
+ Display text and media files (audio, video, images) in the same chat message bubble.
6
+
7
+ ## Installation
8
+
9
+ ```bash
10
+ pip install gradio_multimodalchatbot
11
+ ```
12
+
13
+ ## Usage
14
 
15
  ```python
16
+
17
  import gradio as gr
18
  from gradio_multimodalchatbot import MultimodalChatbot
19
+ from gradio.data_classes import FileData
20
+
21
+ user_msg1 = {"text": "Hello, what is in this image?",
22
+ "files": [{"file": FileData(path="https://gradio-builds.s3.amazonaws.com/diffusion_image/cute_dog.jpg")}]
23
+ }
24
+ bot_msg1 = {"text": "It is a very cute dog",
25
+ "files": []}
26
+
27
+ user_msg2 = {"text": "Describe this audio clip please.",
28
+ "files": [{"file": FileData(path="cantina.wav")}]}
29
+ bot_msg2 = {"text": "It is the cantina song from Star Wars",
30
+ "files": []}
31
+
32
+ user_msg3 = {"text": "Give me a video clip please.",
33
+ "files": []}
34
+ bot_msg3 = {"text": "Here is a video clip of the world",
35
+ "files": [{"file": FileData(path="world.mp4")},
36
+ {"file": FileData(path="cantina.wav")}]}
37
+
38
+ conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
39
+
40
+ with gr.Blocks() as demo:
41
+ MultimodalChatbot(value=conversation, height=800)
42
+
43
+
44
+ if __name__ == "__main__":
45
+ demo.launch()
46
+
47
+ ```
48
+
49
+ ## `MultimodalChatbot`
50
+
51
+ ### Initialization
52
+
53
+ <table>
54
+ <thead>
55
+ <tr>
56
+ <th align="left">name</th>
57
+ <th align="left" style="width: 25%;">type</th>
58
+ <th align="left">default</th>
59
+ <th align="left">description</th>
60
+ </tr>
61
+ </thead>
62
+ <tbody>
63
+ <tr>
64
+ <td align="left"><code>value</code></td>
65
+ <td align="left" style="width: 25%;">
66
+
67
+ ```python
68
+ list[
69
+ list[
70
+ str
71
+ | tuple[str]
72
+ | tuple[str | pathlib.Path, str]
73
+ | None
74
+ ]
75
+ ]
76
+ | Callable
77
+ | None
78
+ ```
79
+
80
+ </td>
81
+ <td align="left"><code>None</code></td>
82
+ <td align="left">Default value to show in chatbot. If callable, the function will be called whenever the app loads to set the initial value of the component.</td>
83
+ </tr>
84
+
85
+ <tr>
86
+ <td align="left"><code>label</code></td>
87
+ <td align="left" style="width: 25%;">
88
+
89
+ ```python
90
+ str | None
91
+ ```
92
+
93
+ </td>
94
+ <td align="left"><code>None</code></td>
95
+ <td align="left">The label for this component. Appears above the component and is also used as the header if there are a table of examples for this component. If None and used in a `gr.Interface`, the label will be the name of the parameter this component is assigned to.</td>
96
+ </tr>
97
+
98
+ <tr>
99
+ <td align="left"><code>every</code></td>
100
+ <td align="left" style="width: 25%;">
101
+
102
+ ```python
103
+ float | None
104
+ ```
105
+
106
+ </td>
107
+ <td align="left"><code>None</code></td>
108
+ <td align="left">If `value` is a callable, run the function 'every' number of seconds while the client connection is open. Has no effect otherwise. Queue must be enabled. The event can be accessed (e.g. to cancel it) via this component's .load_event attribute.</td>
109
+ </tr>
110
+
111
+ <tr>
112
+ <td align="left"><code>show_label</code></td>
113
+ <td align="left" style="width: 25%;">
114
+
115
+ ```python
116
+ bool | None
117
+ ```
118
+
119
+ </td>
120
+ <td align="left"><code>None</code></td>
121
+ <td align="left">if True, will display label.</td>
122
+ </tr>
123
+
124
+ <tr>
125
+ <td align="left"><code>container</code></td>
126
+ <td align="left" style="width: 25%;">
127
+
128
+ ```python
129
+ bool
130
+ ```
131
+
132
+ </td>
133
+ <td align="left"><code>True</code></td>
134
+ <td align="left">If True, will place the component in a container - providing some extra padding around the border.</td>
135
+ </tr>
136
+
137
+ <tr>
138
+ <td align="left"><code>scale</code></td>
139
+ <td align="left" style="width: 25%;">
140
+
141
+ ```python
142
+ int | None
143
+ ```
144
+
145
+ </td>
146
+ <td align="left"><code>None</code></td>
147
+ <td align="left">relative width compared to adjacent Components in a Row. For example, if Component A has scale=2, and Component B has scale=1, A will be twice as wide as B. Should be an integer.</td>
148
+ </tr>
149
+
150
+ <tr>
151
+ <td align="left"><code>min_width</code></td>
152
+ <td align="left" style="width: 25%;">
153
+
154
+ ```python
155
+ int
156
+ ```
157
+
158
+ </td>
159
+ <td align="left"><code>160</code></td>
160
+ <td align="left">minimum pixel width, will wrap if not sufficient screen space to satisfy this value. If a certain scale value results in this Component being narrower than min_width, the min_width parameter will be respected first.</td>
161
+ </tr>
162
+
163
+ <tr>
164
+ <td align="left"><code>visible</code></td>
165
+ <td align="left" style="width: 25%;">
166
+
167
+ ```python
168
+ bool
169
+ ```
170
+
171
+ </td>
172
+ <td align="left"><code>True</code></td>
173
+ <td align="left">If False, component will be hidden.</td>
174
+ </tr>
175
+
176
+ <tr>
177
+ <td align="left"><code>elem_id</code></td>
178
+ <td align="left" style="width: 25%;">
179
+
180
+ ```python
181
+ str | None
182
+ ```
183
+
184
+ </td>
185
+ <td align="left"><code>None</code></td>
186
+ <td align="left">An optional string that is assigned as the id of this component in the HTML DOM. Can be used for targeting CSS styles.</td>
187
+ </tr>
188
+
189
+ <tr>
190
+ <td align="left"><code>elem_classes</code></td>
191
+ <td align="left" style="width: 25%;">
192
+
193
+ ```python
194
+ list[str] | str | None
195
+ ```
196
+
197
+ </td>
198
+ <td align="left"><code>None</code></td>
199
+ <td align="left">An optional list of strings that are assigned as the classes of this component in the HTML DOM. Can be used for targeting CSS styles.</td>
200
+ </tr>
201
+
202
+ <tr>
203
+ <td align="left"><code>render</code></td>
204
+ <td align="left" style="width: 25%;">
205
+
206
+ ```python
207
+ bool
208
+ ```
209
+
210
+ </td>
211
+ <td align="left"><code>True</code></td>
212
+ <td align="left">If False, component will not render be rendered in the Blocks context. Should be used if the intention is to assign event listeners now but render the component later.</td>
213
+ </tr>
214
+
215
+ <tr>
216
+ <td align="left"><code>height</code></td>
217
+ <td align="left" style="width: 25%;">
218
+
219
+ ```python
220
+ int | None
221
+ ```
222
+
223
+ </td>
224
+ <td align="left"><code>None</code></td>
225
+ <td align="left">height of the component in pixels.</td>
226
+ </tr>
227
+
228
+ <tr>
229
+ <td align="left"><code>latex_delimiters</code></td>
230
+ <td align="left" style="width: 25%;">
231
+
232
+ ```python
233
+ list[dict[str, str | bool]] | None
234
+ ```
235
+
236
+ </td>
237
+ <td align="left"><code>None</code></td>
238
+ <td align="left">A list of dicts of the form {"left": open delimiter (str), "right": close delimiter (str), "display": whether to display in newline (bool)} that will be used to render LaTeX expressions. If not provided, `latex_delimiters` is set to `[{ "left": "$$", "right": "$$", "display": True }]`, so only expressions enclosed in $$ delimiters will be rendered as LaTeX, and in a new line. Pass in an empty list to disable LaTeX rendering. For more information, see the [KaTeX documentation](https://katex.org/docs/autorender.html).</td>
239
+ </tr>
240
+
241
+ <tr>
242
+ <td align="left"><code>rtl</code></td>
243
+ <td align="left" style="width: 25%;">
244
+
245
+ ```python
246
+ bool
247
+ ```
248
+
249
+ </td>
250
+ <td align="left"><code>False</code></td>
251
+ <td align="left">If True, sets the direction of the rendered text to right-to-left. Default is False, which renders text left-to-right.</td>
252
+ </tr>
253
+
254
+ <tr>
255
+ <td align="left"><code>show_share_button</code></td>
256
+ <td align="left" style="width: 25%;">
257
+
258
+ ```python
259
+ bool | None
260
+ ```
261
+
262
+ </td>
263
+ <td align="left"><code>None</code></td>
264
+ <td align="left">If True, will show a share icon in the corner of the component that allows user to share outputs to Hugging Face Spaces Discussions. If False, icon does not appear. If set to None (default behavior), then the icon appears if this Gradio app is launched on Spaces, but not otherwise.</td>
265
+ </tr>
266
+
267
+ <tr>
268
+ <td align="left"><code>show_copy_button</code></td>
269
+ <td align="left" style="width: 25%;">
270
+
271
+ ```python
272
+ bool
273
+ ```
274
+
275
+ </td>
276
+ <td align="left"><code>False</code></td>
277
+ <td align="left">If True, will show a copy button for each chatbot message.</td>
278
+ </tr>
279
+
280
+ <tr>
281
+ <td align="left"><code>avatar_images</code></td>
282
+ <td align="left" style="width: 25%;">
283
+
284
+ ```python
285
+ tuple[
286
+ str | pathlib.Path | None, str | pathlib.Path | None
287
+ ]
288
+ | None
289
+ ```
290
+
291
+ </td>
292
+ <td align="left"><code>None</code></td>
293
+ <td align="left">Tuple of two avatar image paths or URLs for user and bot (in that order). Pass None for either the user or bot image to skip. Must be within the working directory of the Gradio app or an external URL.</td>
294
+ </tr>
295
+
296
+ <tr>
297
+ <td align="left"><code>sanitize_html</code></td>
298
+ <td align="left" style="width: 25%;">
299
+
300
+ ```python
301
+ bool
302
+ ```
303
+
304
+ </td>
305
+ <td align="left"><code>True</code></td>
306
+ <td align="left">If False, will disable HTML sanitization for chatbot messages. This is not recommended, as it can lead to security vulnerabilities.</td>
307
+ </tr>
308
+
309
+ <tr>
310
+ <td align="left"><code>render_markdown</code></td>
311
+ <td align="left" style="width: 25%;">
312
+
313
+ ```python
314
+ bool
315
+ ```
316
+
317
+ </td>
318
+ <td align="left"><code>True</code></td>
319
+ <td align="left">If False, will disable Markdown rendering for chatbot messages.</td>
320
+ </tr>
321
+
322
+ <tr>
323
+ <td align="left"><code>bubble_full_width</code></td>
324
+ <td align="left" style="width: 25%;">
325
+
326
+ ```python
327
+ bool
328
+ ```
329
+
330
+ </td>
331
+ <td align="left"><code>True</code></td>
332
+ <td align="left">If False, the chat bubble will fit to the content of the message. If True (default), the chat bubble will be the full width of the component.</td>
333
+ </tr>
334
+
335
+ <tr>
336
+ <td align="left"><code>line_breaks</code></td>
337
+ <td align="left" style="width: 25%;">
338
+
339
+ ```python
340
+ bool
341
+ ```
342
+
343
+ </td>
344
+ <td align="left"><code>True</code></td>
345
+ <td align="left">If True (default), will enable Github-flavored Markdown line breaks in chatbot messages. If False, single new lines will be ignored. Only applies if `render_markdown` is True.</td>
346
+ </tr>
347
+
348
+ <tr>
349
+ <td align="left"><code>likeable</code></td>
350
+ <td align="left" style="width: 25%;">
351
+
352
+ ```python
353
+ bool
354
+ ```
355
+
356
+ </td>
357
+ <td align="left"><code>False</code></td>
358
+ <td align="left">Whether the chat messages display a like or dislike button. Set automatically by the .like method but has to be present in the signature for it to show up in the config.</td>
359
+ </tr>
360
+
361
+ <tr>
362
+ <td align="left"><code>layout</code></td>
363
+ <td align="left" style="width: 25%;">
364
+
365
+ ```python
366
+ "panel" | "bubble" | None
367
+ ```
368
+
369
+ </td>
370
+ <td align="left"><code>None</code></td>
371
+ <td align="left">If "panel", will display the chatbot in a llm style layout. If "bubble", will display the chatbot with message bubbles, with the user and bot messages on alterating sides. Will default to "bubble".</td>
372
+ </tr>
373
+ </tbody></table>
374
+
375
+
376
+ ### Events
377
+
378
+ | name | description |
379
+ |:-----|:------------|
380
+ | `change` | Triggered when the value of the MultimodalChatbot changes either because of user input (e.g. a user types in a textbox) OR because of a function update (e.g. an image receives a value from the output of an event trigger). See `.input()` for a listener that is only triggered by user input. |
381
+ | `select` | Event listener for when the user selects or deselects the MultimodalChatbot. Uses event data gradio.SelectData to carry `value` referring to the label of the MultimodalChatbot, and `selected` to refer to state of the MultimodalChatbot. See EventData documentation on how to use this event data |
382
+ | `like` | This listener is triggered when the user likes/dislikes from within the MultimodalChatbot. This event has EventData of type gradio.LikeData that carries information, accessible through LikeData.index and LikeData.value. See EventData documentation on how to use this event data. |
383
+
384
+
385
+
386
+ ### User function
387
+
388
+ The impact on the users predict function varies depending on whether the component is used as an input or output for an event (or both).
389
+
390
+ - When used as an Input, the component only impacts the input signature of the user function.
391
+ - When used as an output, the component only impacts the return signature of the user function.
392
+
393
+ The code snippet below is accurate in cases where the component is used as both an input and an output.
394
+
395
+
396
+
397
+ ```python
398
+ def predict(
399
+ value: list[MultimodalMessage] | None
400
+ ) -> list[
401
+ list[str | tuple[str] | tuple[str, str] | None] | tuple
402
+ ]:
403
+ return value
404
+ ```
405
+
406
+
407
+ ## `MultimodalMessage`
408
+ ```python
409
+ class MultimodalMessage(GradioModel):
410
+ text: Optional[str] = None
411
+ files: Optional[List[FileMessage]] = None
412
+ ```
413
+
414
+ ## `FileMessage`
415
+ ```python
416
+ class FileMessage(GradioModel):
417
+ file: FileData
418
+ alt_text: Optional[str] = None
419
  ```
src/backend/gradio_multimodalchatbot/multimodalchatbot.py CHANGED
@@ -144,7 +144,7 @@ class MultimodalChatbot(Component):
144
  return None
145
  if isinstance(chat_message, dict):
146
  chat_message = MultimodalMessage(**chat_message)
147
- chat_message.text = inspect.cleandoc(chat_message.text)
148
  for file_ in chat_message.files:
149
  file_.file.mime_type = client_utils.get_mimetype(file_.file.path)
150
  return chat_message
 
144
  return None
145
  if isinstance(chat_message, dict):
146
  chat_message = MultimodalMessage(**chat_message)
147
+ chat_message.text = inspect.cleandoc(chat_message.text or "")
148
  for file_ in chat_message.files:
149
  file_.file.mime_type = client_utils.get_mimetype(file_.file.path)
150
  return chat_message
src/backend/gradio_multimodalchatbot/multimodalchatbot.pyi CHANGED
@@ -144,7 +144,7 @@ class MultimodalChatbot(Component):
144
  return None
145
  if isinstance(chat_message, dict):
146
  chat_message = MultimodalMessage(**chat_message)
147
- chat_message.text = inspect.cleandoc(chat_message.text)
148
  for file_ in chat_message.files:
149
  file_.file.mime_type = client_utils.get_mimetype(file_.file.path)
150
  return chat_message
@@ -182,7 +182,6 @@ class MultimodalChatbot(Component):
182
  inputs: Component | Sequence[Component] | set[Component] | None = None,
183
  outputs: Component | Sequence[Component] | None = None,
184
  api_name: str | None | Literal[False] = None,
185
- status_tracker: None = None,
186
  scroll_to_output: bool = False,
187
  show_progress: Literal["full", "minimal", "hidden"] = "full",
188
  queue: bool | None = None,
@@ -193,7 +192,10 @@ class MultimodalChatbot(Component):
193
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
194
  every: float | None = None,
195
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
196
- js: str | None = None,) -> Dependency:
 
 
 
197
  """
198
  Parameters:
199
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
@@ -211,6 +213,9 @@ class MultimodalChatbot(Component):
211
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
212
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
213
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
 
 
 
214
  """
215
  ...
216
 
@@ -219,7 +224,6 @@ class MultimodalChatbot(Component):
219
  inputs: Component | Sequence[Component] | set[Component] | None = None,
220
  outputs: Component | Sequence[Component] | None = None,
221
  api_name: str | None | Literal[False] = None,
222
- status_tracker: None = None,
223
  scroll_to_output: bool = False,
224
  show_progress: Literal["full", "minimal", "hidden"] = "full",
225
  queue: bool | None = None,
@@ -230,7 +234,10 @@ class MultimodalChatbot(Component):
230
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
231
  every: float | None = None,
232
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
233
- js: str | None = None,) -> Dependency:
 
 
 
234
  """
235
  Parameters:
236
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
@@ -248,6 +255,9 @@ class MultimodalChatbot(Component):
248
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
249
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
250
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
 
 
 
251
  """
252
  ...
253
 
@@ -256,7 +266,6 @@ class MultimodalChatbot(Component):
256
  inputs: Component | Sequence[Component] | set[Component] | None = None,
257
  outputs: Component | Sequence[Component] | None = None,
258
  api_name: str | None | Literal[False] = None,
259
- status_tracker: None = None,
260
  scroll_to_output: bool = False,
261
  show_progress: Literal["full", "minimal", "hidden"] = "full",
262
  queue: bool | None = None,
@@ -267,7 +276,10 @@ class MultimodalChatbot(Component):
267
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
268
  every: float | None = None,
269
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
270
- js: str | None = None,) -> Dependency:
 
 
 
271
  """
272
  Parameters:
273
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
@@ -285,5 +297,8 @@ class MultimodalChatbot(Component):
285
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
286
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
287
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
 
 
 
288
  """
289
  ...
 
144
  return None
145
  if isinstance(chat_message, dict):
146
  chat_message = MultimodalMessage(**chat_message)
147
+ chat_message.text = inspect.cleandoc(chat_message.text or "")
148
  for file_ in chat_message.files:
149
  file_.file.mime_type = client_utils.get_mimetype(file_.file.path)
150
  return chat_message
 
182
  inputs: Component | Sequence[Component] | set[Component] | None = None,
183
  outputs: Component | Sequence[Component] | None = None,
184
  api_name: str | None | Literal[False] = None,
 
185
  scroll_to_output: bool = False,
186
  show_progress: Literal["full", "minimal", "hidden"] = "full",
187
  queue: bool | None = None,
 
192
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
193
  every: float | None = None,
194
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
195
+ js: str | None = None,
196
+ concurrency_limit: int | None | Literal["default"] = "default",
197
+ concurrency_id: str | None = None,
198
+ show_api: bool = True) -> Dependency:
199
  """
200
  Parameters:
201
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
 
213
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
214
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
215
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
216
+ concurrency_limit: If set, this is the maximum number of this event that can be running simultaneously. Can be set to None to mean no concurrency_limit (any number of this event can be running simultaneously). Set to "default" to use the default concurrency limit (defined by the `default_concurrency_limit` parameter in `Blocks.queue()`, which itself is 1 by default).
217
+ concurrency_id: If set, this is the id of the concurrency group. Events with the same concurrency_id will be limited by the lowest set concurrency_limit.
218
+ show_api: whether to show this event in the "view API" page of the Gradio app, or in the ".view_api()" method of the Gradio clients. Unlike setting api_name to False, setting show_api to False will still allow downstream apps to use this event. If fn is None, show_api will automatically be set to False.
219
  """
220
  ...
221
 
 
224
  inputs: Component | Sequence[Component] | set[Component] | None = None,
225
  outputs: Component | Sequence[Component] | None = None,
226
  api_name: str | None | Literal[False] = None,
 
227
  scroll_to_output: bool = False,
228
  show_progress: Literal["full", "minimal", "hidden"] = "full",
229
  queue: bool | None = None,
 
234
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
235
  every: float | None = None,
236
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
237
+ js: str | None = None,
238
+ concurrency_limit: int | None | Literal["default"] = "default",
239
+ concurrency_id: str | None = None,
240
+ show_api: bool = True) -> Dependency:
241
  """
242
  Parameters:
243
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
 
255
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
256
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
257
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
258
+ concurrency_limit: If set, this is the maximum number of this event that can be running simultaneously. Can be set to None to mean no concurrency_limit (any number of this event can be running simultaneously). Set to "default" to use the default concurrency limit (defined by the `default_concurrency_limit` parameter in `Blocks.queue()`, which itself is 1 by default).
259
+ concurrency_id: If set, this is the id of the concurrency group. Events with the same concurrency_id will be limited by the lowest set concurrency_limit.
260
+ show_api: whether to show this event in the "view API" page of the Gradio app, or in the ".view_api()" method of the Gradio clients. Unlike setting api_name to False, setting show_api to False will still allow downstream apps to use this event. If fn is None, show_api will automatically be set to False.
261
  """
262
  ...
263
 
 
266
  inputs: Component | Sequence[Component] | set[Component] | None = None,
267
  outputs: Component | Sequence[Component] | None = None,
268
  api_name: str | None | Literal[False] = None,
 
269
  scroll_to_output: bool = False,
270
  show_progress: Literal["full", "minimal", "hidden"] = "full",
271
  queue: bool | None = None,
 
276
  cancels: dict[str, Any] | list[dict[str, Any]] | None = None,
277
  every: float | None = None,
278
  trigger_mode: Literal["once", "multiple", "always_last"] | None = None,
279
+ js: str | None = None,
280
+ concurrency_limit: int | None | Literal["default"] = "default",
281
+ concurrency_id: str | None = None,
282
+ show_api: bool = True) -> Dependency:
283
  """
284
  Parameters:
285
  fn: the function to call when this event is triggered. Often a machine learning model's prediction function. Each parameter of the function corresponds to one input component, and the function should return a single value or a tuple of values, with each element in the tuple corresponding to one output component.
 
297
  every: Run this event 'every' number of seconds while the client connection is open. Interpreted in seconds. Queue must be enabled.
298
  trigger_mode: If "once" (default for all events except `.change()`) would not allow any submissions while an event is pending. If set to "multiple", unlimited submissions are allowed while pending, and "always_last" (default for `.change()` event) would allow a second submission after the pending event is complete.
299
  js: Optional frontend js method to run before running 'fn'. Input arguments for js method are values of 'inputs' and 'outputs', return should be a list of values for output components.
300
+ concurrency_limit: If set, this is the maximum number of this event that can be running simultaneously. Can be set to None to mean no concurrency_limit (any number of this event can be running simultaneously). Set to "default" to use the default concurrency limit (defined by the `default_concurrency_limit` parameter in `Blocks.queue()`, which itself is 1 by default).
301
+ concurrency_id: If set, this is the id of the concurrency group. Events with the same concurrency_id will be limited by the lowest set concurrency_limit.
302
+ show_api: whether to show this event in the "view API" page of the Gradio app, or in the ".view_api()" method of the Gradio clients. Unlike setting api_name to False, setting show_api to False will still allow downstream apps to use this event. If fn is None, show_api will automatically be set to False.
303
  """
304
  ...
src/demo/app.py CHANGED
@@ -23,7 +23,8 @@ bot_msg3 = {"text": "Here is a video clip of the world",
23
  conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
24
 
25
  with gr.Blocks() as demo:
26
- MultimodalChatbot(value=conversation)
27
 
28
 
29
- demo.launch()
 
 
23
  conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
24
 
25
  with gr.Blocks() as demo:
26
+ MultimodalChatbot(value=conversation, height=800)
27
 
28
 
29
+ if __name__ == "__main__":
30
+ demo.launch()
src/demo/css.css ADDED
@@ -0,0 +1,157 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ html {
2
+ font-family: Inter;
3
+ font-size: 16px;
4
+ font-weight: 400;
5
+ line-height: 1.5;
6
+ -webkit-text-size-adjust: 100%;
7
+ background: #fff;
8
+ color: #323232;
9
+ -webkit-font-smoothing: antialiased;
10
+ -moz-osx-font-smoothing: grayscale;
11
+ text-rendering: optimizeLegibility;
12
+ }
13
+
14
+ :root {
15
+ --space: 1;
16
+ --vspace: calc(var(--space) * 1rem);
17
+ --vspace-0: calc(3 * var(--space) * 1rem);
18
+ --vspace-1: calc(2 * var(--space) * 1rem);
19
+ --vspace-2: calc(1.5 * var(--space) * 1rem);
20
+ --vspace-3: calc(0.5 * var(--space) * 1rem);
21
+ }
22
+
23
+ .app {
24
+ max-width: 748px !important;
25
+ }
26
+
27
+ .prose p {
28
+ margin: var(--vspace) 0;
29
+ line-height: var(--vspace * 2);
30
+ font-size: 1rem;
31
+ }
32
+
33
+ code {
34
+ font-family: "Inconsolata", sans-serif;
35
+ font-size: 16px;
36
+ }
37
+
38
+ h1,
39
+ h1 code {
40
+ font-weight: 400;
41
+ line-height: calc(2.5 / var(--space) * var(--vspace));
42
+ }
43
+
44
+ h1 code {
45
+ background: none;
46
+ border: none;
47
+ letter-spacing: 0.05em;
48
+ padding-bottom: 5px;
49
+ position: relative;
50
+ padding: 0;
51
+ }
52
+
53
+ h2 {
54
+ margin: var(--vspace-1) 0 var(--vspace-2) 0;
55
+ line-height: 1em;
56
+ }
57
+
58
+ h3,
59
+ h3 code {
60
+ margin: var(--vspace-1) 0 var(--vspace-2) 0;
61
+ line-height: 1em;
62
+ }
63
+
64
+ h4,
65
+ h5,
66
+ h6 {
67
+ margin: var(--vspace-3) 0 var(--vspace-3) 0;
68
+ line-height: var(--vspace);
69
+ }
70
+
71
+ .bigtitle,
72
+ h1,
73
+ h1 code {
74
+ font-size: calc(8px * 4.5);
75
+ word-break: break-word;
76
+ }
77
+
78
+ .title,
79
+ h2,
80
+ h2 code {
81
+ font-size: calc(8px * 3.375);
82
+ font-weight: lighter;
83
+ word-break: break-word;
84
+ border: none;
85
+ background: none;
86
+ }
87
+
88
+ .subheading1,
89
+ h3,
90
+ h3 code {
91
+ font-size: calc(8px * 1.8);
92
+ font-weight: 600;
93
+ border: none;
94
+ background: none;
95
+ letter-spacing: 0.1em;
96
+ text-transform: uppercase;
97
+ }
98
+
99
+ h2 code {
100
+ padding: 0;
101
+ position: relative;
102
+ letter-spacing: 0.05em;
103
+ }
104
+
105
+ blockquote {
106
+ font-size: calc(8px * 1.1667);
107
+ font-style: italic;
108
+ line-height: calc(1.1667 * var(--vspace));
109
+ margin: var(--vspace-2) var(--vspace-2);
110
+ }
111
+
112
+ .subheading2,
113
+ h4 {
114
+ font-size: calc(8px * 1.4292);
115
+ text-transform: uppercase;
116
+ font-weight: 600;
117
+ }
118
+
119
+ .subheading3,
120
+ h5 {
121
+ font-size: calc(8px * 1.2917);
122
+ line-height: calc(1.2917 * var(--vspace));
123
+
124
+ font-weight: lighter;
125
+ text-transform: uppercase;
126
+ letter-spacing: 0.15em;
127
+ }
128
+
129
+ h6 {
130
+ font-size: calc(8px * 1.1667);
131
+ font-size: 1.1667em;
132
+ font-weight: normal;
133
+ font-style: italic;
134
+ font-family: "le-monde-livre-classic-byol", serif !important;
135
+ letter-spacing: 0px !important;
136
+ }
137
+
138
+ #start .md > *:first-child {
139
+ margin-top: 0;
140
+ }
141
+
142
+ h2 + h3 {
143
+ margin-top: 0;
144
+ }
145
+
146
+ .md hr {
147
+ border: none;
148
+ border-top: 1px solid var(--block-border-color);
149
+ margin: var(--vspace-2) 0 var(--vspace-2) 0;
150
+ }
151
+ .prose ul {
152
+ margin: var(--vspace-2) 0 var(--vspace-1) 0;
153
+ }
154
+
155
+ .gap {
156
+ gap: 0;
157
+ }
src/demo/space.py ADDED
@@ -0,0 +1,171 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+
2
+ import gradio as gr
3
+ from app import demo as app
4
+ import os
5
+
6
+ _docs = {'MultimodalChatbot': {'description': 'Displays a chatbot output showing both user submitted messages and responses. Supports a subset of Markdown including bold, italics, code, tables. Also supports audio/video/image files, which are displayed in the MultimodalChatbot, and other kinds of files which are displayed as links.\n', 'members': {'__init__': {'value': {'type': 'list[\n list[\n str\n | tuple[str]\n | tuple[str | pathlib.Path, str]\n | None\n ]\n ]\n | Callable\n | None', 'default': 'None', 'description': 'Default value to show in chatbot. If callable, the function will be called whenever the app loads to set the initial value of the component.'}, 'label': {'type': 'str | None', 'default': 'None', 'description': 'The label for this component. Appears above the component and is also used as the header if there are a table of examples for this component. If None and used in a `gr.Interface`, the label will be the name of the parameter this component is assigned to.'}, 'every': {'type': 'float | None', 'default': 'None', 'description': "If `value` is a callable, run the function 'every' number of seconds while the client connection is open. Has no effect otherwise. Queue must be enabled. The event can be accessed (e.g. to cancel it) via this component's .load_event attribute."}, 'show_label': {'type': 'bool | None', 'default': 'None', 'description': 'if True, will display label.'}, 'container': {'type': 'bool', 'default': 'True', 'description': 'If True, will place the component in a container - providing some extra padding around the border.'}, 'scale': {'type': 'int | None', 'default': 'None', 'description': 'relative width compared to adjacent Components in a Row. For example, if Component A has scale=2, and Component B has scale=1, A will be twice as wide as B. Should be an integer.'}, 'min_width': {'type': 'int', 'default': '160', 'description': 'minimum pixel width, will wrap if not sufficient screen space to satisfy this value. If a certain scale value results in this Component being narrower than min_width, the min_width parameter will be respected first.'}, 'visible': {'type': 'bool', 'default': 'True', 'description': 'If False, component will be hidden.'}, 'elem_id': {'type': 'str | None', 'default': 'None', 'description': 'An optional string that is assigned as the id of this component in the HTML DOM. Can be used for targeting CSS styles.'}, 'elem_classes': {'type': 'list[str] | str | None', 'default': 'None', 'description': 'An optional list of strings that are assigned as the classes of this component in the HTML DOM. Can be used for targeting CSS styles.'}, 'render': {'type': 'bool', 'default': 'True', 'description': 'If False, component will not render be rendered in the Blocks context. Should be used if the intention is to assign event listeners now but render the component later.'}, 'height': {'type': 'int | None', 'default': 'None', 'description': 'height of the component in pixels.'}, 'latex_delimiters': {'type': 'list[dict[str, str | bool]] | None', 'default': 'None', 'description': 'A list of dicts of the form {"left": open delimiter (str), "right": close delimiter (str), "display": whether to display in newline (bool)} that will be used to render LaTeX expressions. If not provided, `latex_delimiters` is set to `[{ "left": "$$", "right": "$$", "display": True }]`, so only expressions enclosed in $$ delimiters will be rendered as LaTeX, and in a new line. Pass in an empty list to disable LaTeX rendering. For more information, see the [KaTeX documentation](https://katex.org/docs/autorender.html).'}, 'rtl': {'type': 'bool', 'default': 'False', 'description': 'If True, sets the direction of the rendered text to right-to-left. Default is False, which renders text left-to-right.'}, 'show_share_button': {'type': 'bool | None', 'default': 'None', 'description': 'If True, will show a share icon in the corner of the component that allows user to share outputs to Hugging Face Spaces Discussions. If False, icon does not appear. If set to None (default behavior), then the icon appears if this Gradio app is launched on Spaces, but not otherwise.'}, 'show_copy_button': {'type': 'bool', 'default': 'False', 'description': 'If True, will show a copy button for each chatbot message.'}, 'avatar_images': {'type': 'tuple[\n str | pathlib.Path | None, str | pathlib.Path | None\n ]\n | None', 'default': 'None', 'description': 'Tuple of two avatar image paths or URLs for user and bot (in that order). Pass None for either the user or bot image to skip. Must be within the working directory of the Gradio app or an external URL.'}, 'sanitize_html': {'type': 'bool', 'default': 'True', 'description': 'If False, will disable HTML sanitization for chatbot messages. This is not recommended, as it can lead to security vulnerabilities.'}, 'render_markdown': {'type': 'bool', 'default': 'True', 'description': 'If False, will disable Markdown rendering for chatbot messages.'}, 'bubble_full_width': {'type': 'bool', 'default': 'True', 'description': 'If False, the chat bubble will fit to the content of the message. If True (default), the chat bubble will be the full width of the component.'}, 'line_breaks': {'type': 'bool', 'default': 'True', 'description': 'If True (default), will enable Github-flavored Markdown line breaks in chatbot messages. If False, single new lines will be ignored. Only applies if `render_markdown` is True.'}, 'likeable': {'type': 'bool', 'default': 'False', 'description': 'Whether the chat messages display a like or dislike button. Set automatically by the .like method but has to be present in the signature for it to show up in the config.'}, 'layout': {'type': '"panel" | "bubble" | None', 'default': 'None', 'description': 'If "panel", will display the chatbot in a llm style layout. If "bubble", will display the chatbot with message bubbles, with the user and bot messages on alterating sides. Will default to "bubble".'}}, 'postprocess': {'value': {'type': 'list[\n list[str | tuple[str] | tuple[str, str] | None] | tuple\n]', 'description': None}}, 'preprocess': {'return': {'type': 'list[MultimodalMessage] | None', 'description': None}, 'value': None}}, 'events': {'change': {'type': None, 'default': None, 'description': 'Triggered when the value of the MultimodalChatbot changes either because of user input (e.g. a user types in a textbox) OR because of a function update (e.g. an image receives a value from the output of an event trigger). See `.input()` for a listener that is only triggered by user input.'}, 'select': {'type': None, 'default': None, 'description': 'Event listener for when the user selects or deselects the MultimodalChatbot. Uses event data gradio.SelectData to carry `value` referring to the label of the MultimodalChatbot, and `selected` to refer to state of the MultimodalChatbot. See EventData documentation on how to use this event data'}, 'like': {'type': None, 'default': None, 'description': 'This listener is triggered when the user likes/dislikes from within the MultimodalChatbot. This event has EventData of type gradio.LikeData that carries information, accessible through LikeData.index and LikeData.value. See EventData documentation on how to use this event data.'}}}, '__meta__': {'additional_interfaces': {'MultimodalMessage': {'source': 'class MultimodalMessage(GradioModel):\n text: Optional[str] = None\n files: Optional[List[FileMessage]] = None', 'refs': ['FileMessage']}, 'FileMessage': {'source': 'class FileMessage(GradioModel):\n file: FileData\n alt_text: Optional[str] = None'}}, 'user_fn_refs': {'MultimodalChatbot': ['MultimodalMessage']}}}
7
+
8
+ abs_path = os.path.join(os.path.dirname(__file__), "css.css")
9
+
10
+ with gr.Blocks(
11
+ css=abs_path,
12
+ theme=gr.themes.Default(
13
+ font_mono=[
14
+ gr.themes.GoogleFont("Inconsolata"),
15
+ "monospace",
16
+ ],
17
+ ),
18
+ ) as demo:
19
+ gr.Markdown(
20
+ """
21
+ # `gradio_multimodalchatbot`
22
+
23
+ <div style="display: flex; gap: 7px;">
24
+ <a href="https://pypi.org/project/gradio_multimodalchatbot/" target="_blank"><img alt="PyPI - Version" src="https://img.shields.io/pypi/v/gradio_multimodalchatbot"></a> <a href="https://huggingface.co/spaces/freddyaboulton/gradio_multimodalchatbot//discussions" target="_blank"><img alt="Static Badge" src="https://img.shields.io/badge/%F0%9F%A4%97%20Discuss-%23097EFF?style=flat&logoColor=black"></a>
25
+ </div>
26
+
27
+ Display text and media files (audio, video, images) in the same chat message bubble.
28
+ """, elem_classes=["md-custom"], header_links=True)
29
+ app.render()
30
+ gr.Markdown(
31
+ """
32
+ ## Installation
33
+
34
+ ```bash
35
+ pip install gradio_multimodalchatbot
36
+ ```
37
+
38
+ ## Usage
39
+
40
+ ```python
41
+
42
+ import gradio as gr
43
+ from gradio_multimodalchatbot import MultimodalChatbot
44
+ from gradio.data_classes import FileData
45
+
46
+ user_msg1 = {"text": "Hello, what is in this image?",
47
+ "files": [{"file": FileData(path="https://gradio-builds.s3.amazonaws.com/diffusion_image/cute_dog.jpg")}]
48
+ }
49
+ bot_msg1 = {"text": "It is a very cute dog",
50
+ "files": []}
51
+
52
+ user_msg2 = {"text": "Describe this audio clip please.",
53
+ "files": [{"file": FileData(path="cantina.wav")}]}
54
+ bot_msg2 = {"text": "It is the cantina song from Star Wars",
55
+ "files": []}
56
+
57
+ user_msg3 = {"text": "Give me a video clip please.",
58
+ "files": []}
59
+ bot_msg3 = {"text": "Here is a video clip of the world",
60
+ "files": [{"file": FileData(path="world.mp4")},
61
+ {"file": FileData(path="cantina.wav")}]}
62
+
63
+ conversation = [[user_msg1, bot_msg1], [user_msg2, bot_msg2], [user_msg3, bot_msg3]]
64
+
65
+ with gr.Blocks() as demo:
66
+ MultimodalChatbot(value=conversation, height=800)
67
+
68
+
69
+ if __name__ == "__main__":
70
+ demo.launch()
71
+
72
+ ```
73
+ """, elem_classes=["md-custom"], header_links=True)
74
+
75
+
76
+ gr.Markdown("""
77
+ ## `MultimodalChatbot`
78
+
79
+ ### Initialization
80
+ """, elem_classes=["md-custom"], header_links=True)
81
+
82
+ gr.ParamViewer(value=_docs["MultimodalChatbot"]["members"]["__init__"], linkify=['MultimodalMessage', 'FileMessage'])
83
+
84
+
85
+ gr.Markdown("### Events")
86
+ gr.ParamViewer(value=_docs["MultimodalChatbot"]["events"], linkify=['Event'])
87
+
88
+
89
+
90
+
91
+ gr.Markdown("""
92
+
93
+ ### User function
94
+
95
+ The impact on the users predict function varies depending on whether the component is used as an input or output for an event (or both).
96
+
97
+ - When used as an Input, the component only impacts the input signature of the user function.
98
+ - When used as an output, the component only impacts the return signature of the user function.
99
+
100
+ The code snippet below is accurate in cases where the component is used as both an input and an output.
101
+
102
+
103
+
104
+ ```python
105
+ def predict(
106
+ value: list[MultimodalMessage] | None
107
+ ) -> list[
108
+ list[str | tuple[str] | tuple[str, str] | None] | tuple
109
+ ]:
110
+ return value
111
+ ```
112
+ """, elem_classes=["md-custom", "MultimodalChatbot-user-fn"], header_links=True)
113
+
114
+
115
+
116
+
117
+ code_MultimodalMessage = gr.Markdown("""
118
+ ## `MultimodalMessage`
119
+ ```python
120
+ class MultimodalMessage(GradioModel):
121
+ text: Optional[str] = None
122
+ files: Optional[List[FileMessage]] = None
123
+ ```""", elem_classes=["md-custom", "MultimodalMessage"], header_links=True)
124
+
125
+ code_FileMessage = gr.Markdown("""
126
+ ## `FileMessage`
127
+ ```python
128
+ class FileMessage(GradioModel):
129
+ file: FileData
130
+ alt_text: Optional[str] = None
131
+ ```""", elem_classes=["md-custom", "FileMessage"], header_links=True)
132
+
133
+ demo.load(None, js=r"""function() {
134
+ const refs = {
135
+ MultimodalMessage: ['FileMessage'],
136
+ FileMessage: [], };
137
+ const user_fn_refs = {
138
+ MultimodalChatbot: ['MultimodalMessage'], };
139
+ requestAnimationFrame(() => {
140
+
141
+ Object.entries(user_fn_refs).forEach(([key, refs]) => {
142
+ if (refs.length > 0) {
143
+ const el = document.querySelector(`.${key}-user-fn`);
144
+ if (!el) return;
145
+ refs.forEach(ref => {
146
+ el.innerHTML = el.innerHTML.replace(
147
+ new RegExp("\\b"+ref+"\\b", "g"),
148
+ `<a href="#h-${ref.toLowerCase()}">${ref}</a>`
149
+ );
150
+ })
151
+ }
152
+ })
153
+
154
+ Object.entries(refs).forEach(([key, refs]) => {
155
+ if (refs.length > 0) {
156
+ const el = document.querySelector(`.${key}`);
157
+ if (!el) return;
158
+ refs.forEach(ref => {
159
+ el.innerHTML = el.innerHTML.replace(
160
+ new RegExp("\\b"+ref+"\\b", "g"),
161
+ `<a href="#h-${ref.toLowerCase()}">${ref}</a>`
162
+ );
163
+ })
164
+ }
165
+ })
166
+ })
167
+ }
168
+
169
+ """)
170
+
171
+ demo.launch()
src/pyproject.toml CHANGED
@@ -9,16 +9,15 @@ build-backend = "hatchling.build"
9
  [project]
10
  name = "gradio_multimodalchatbot"
11
  version = "0.0.1"
12
- description = "Python library for easily interacting with trained machine learning models"
13
  readme = "README.md"
14
  license = "Apache-2.0"
15
  requires-python = ">=3.8"
16
  authors = [{ name = "Freddy Boulton", email = "freddy@huggingface.co" }]
17
  keywords = [
18
- "machine learning",
19
- "reproducibility",
20
- "visualization",
21
- "gradio",
22
  "gradio custom component",
23
  "gradio-template-Chatbot"
24
  ]
@@ -42,6 +41,9 @@ classifiers = [
42
  [project.optional-dependencies]
43
  dev = ["build", "twine"]
44
 
 
 
 
45
  [tool.hatch.build]
46
  artifacts = ["/backend/gradio_multimodalchatbot/templates", "*.pyi", "backend/gradio_multimodalchatbot/templates"]
47
 
 
9
  [project]
10
  name = "gradio_multimodalchatbot"
11
  version = "0.0.1"
12
+ description = "Display text and media files (audio, video, images) in the same chat message bubble."
13
  readme = "README.md"
14
  license = "Apache-2.0"
15
  requires-python = ">=3.8"
16
  authors = [{ name = "Freddy Boulton", email = "freddy@huggingface.co" }]
17
  keywords = [
18
+ "Chatbot",
19
+ "Multimodal",
20
+ "Media",
 
21
  "gradio custom component",
22
  "gradio-template-Chatbot"
23
  ]
 
41
  [project.optional-dependencies]
42
  dev = ["build", "twine"]
43
 
44
+ [project.urls]
45
+ space = "https://huggingface.co/spaces/freddyaboulton/gradio_multimodalchatbot/"
46
+
47
  [tool.hatch.build]
48
  artifacts = ["/backend/gradio_multimodalchatbot/templates", "*.pyi", "backend/gradio_multimodalchatbot/templates"]
49