diff --git a/demo/uploader.py b/demo/uploader.py
index 80e298ec..87edad9f 100644
--- a/demo/uploader.py
+++ b/demo/uploader.py
@@ -22,14 +22,43 @@ def load(e: me.LoadEvent):
def app():
state = me.state(State)
with me.box(style=me.Style(padding=me.Padding.all(15))):
- me.uploader(
- label="Upload Image",
- accepted_file_types=["image/jpeg", "image/png"],
- on_upload=handle_upload,
- type="flat",
- color="primary",
- style=me.Style(font_weight="bold"),
- )
+ with me.box(style=me.Style(display="flex", gap=20)):
+ with me.content_uploader(
+ accepted_file_types=["image/jpeg", "image/png"],
+ on_upload=handle_upload,
+ type="flat",
+ color="primary",
+ style=me.Style(font_weight="bold"),
+ ):
+ with me.box(style=me.Style(display="flex", gap=5)):
+ me.icon("upload")
+ me.text("Upload Image", style=me.Style(line_height="25px"))
+
+ with me.content_uploader(
+ accepted_file_types=["image/jpeg", "image/png"],
+ on_upload=handle_upload,
+ type="flat",
+ color="warn",
+ style=me.Style(font_weight="bold"),
+ ):
+ me.icon("upload")
+
+ me.uploader(
+ label="Upload Image",
+ accepted_file_types=["image/jpeg", "image/png"],
+ on_upload=handle_upload,
+ type="flat",
+ color="accent",
+ style=me.Style(font_weight="bold"),
+ )
+
+ with me.content_uploader(
+ accepted_file_types=["image/jpeg", "image/png"],
+ on_upload=handle_upload,
+ type="icon",
+ style=me.Style(font_weight="bold"),
+ ):
+ me.icon("upload")
if state.file.size:
with me.box(style=me.Style(margin=me.Margin.all(10))):
diff --git a/docs/components/uploader.md b/docs/components/uploader.md
index bf71ee35..36269dab 100644
--- a/docs/components/uploader.md
+++ b/docs/components/uploader.md
@@ -14,5 +14,6 @@ matches the look of Angular Material Components.
## API
::: mesop.components.uploader.uploader.uploader
+::: mesop.components.uploader.uploader.content_uploader
::: mesop.components.uploader.uploader.UploadEvent
::: mesop.components.uploader.uploader.UploadedFile
diff --git a/mesop/__init__.py b/mesop/__init__.py
index 44d2e591..38cd9705 100644
--- a/mesop/__init__.py
+++ b/mesop/__init__.py
@@ -179,6 +179,9 @@
from mesop.components.uploader.uploader import (
UploadEvent as UploadEvent,
)
+from mesop.components.uploader.uploader import (
+ content_uploader as content_uploader,
+)
from mesop.components.uploader.uploader import uploader as uploader
from mesop.components.video.video import video as video
from mesop.dataclass_utils import dataclass_with_defaults
diff --git a/mesop/components/uploader/BUILD b/mesop/components/uploader/BUILD
index f2479ee3..ec5b918d 100644
--- a/mesop/components/uploader/BUILD
+++ b/mesop/components/uploader/BUILD
@@ -8,7 +8,10 @@ package(
mesop_component(
name = "uploader",
assets = [":uploader.css"],
- py_deps = [":uploaded_file"],
+ py_deps = [
+ ":uploaded_file",
+ "//mesop/components/text:py",
+ ],
)
sass_binary(
diff --git a/mesop/components/uploader/e2e/__init__.py b/mesop/components/uploader/e2e/__init__.py
index 375243fb..7aa5628a 100644
--- a/mesop/components/uploader/e2e/__init__.py
+++ b/mesop/components/uploader/e2e/__init__.py
@@ -1 +1,2 @@
+from . import content_uploader_app as content_uploader_app
from . import uploader_app as uploader_app
diff --git a/mesop/components/uploader/e2e/content_uploader_app.py b/mesop/components/uploader/e2e/content_uploader_app.py
new file mode 100644
index 00000000..d5a0d098
--- /dev/null
+++ b/mesop/components/uploader/e2e/content_uploader_app.py
@@ -0,0 +1,47 @@
+import base64
+
+import mesop as me
+
+
+@me.stateclass
+class State:
+ file: me.UploadedFile
+ upload_count: int = 0
+
+
+@me.page(path="/components/uploader/e2e/content_uploader_app")
+def app():
+ state = me.state(State)
+ with me.box(style=me.Style(padding=me.Padding.all(15))):
+ with me.content_uploader(
+ accepted_file_types=["image/jpeg", "image/png"],
+ on_upload=handle_upload,
+ type="flat",
+ color="primary",
+ style=me.Style(font_weight="bold"),
+ ):
+ with me.box(style=me.Style(display="flex", gap=5)):
+ me.icon("upload")
+ me.text("Upload Image", style=me.Style(line_height="25px"))
+
+ if state.file.size:
+ with me.box(style=me.Style(margin=me.Margin.all(10))):
+ me.text(f"File name: {state.file.name}")
+ me.text(f"File size: {state.file.size}")
+ me.text(f"File type: {state.file.mime_type}")
+ me.text(f"Upload count: {state.upload_count}")
+
+ with me.box(style=me.Style(margin=me.Margin.all(10))):
+ me.image(src=_convert_contents_data_url(state.file))
+
+
+def handle_upload(event: me.UploadEvent):
+ state = me.state(State)
+ state.file = event.file
+ state.upload_count += 1
+
+
+def _convert_contents_data_url(file: me.UploadedFile) -> str:
+ return (
+ f"data:{file.mime_type};base64,{base64.b64encode(file.getvalue()).decode()}"
+ )
diff --git a/mesop/components/uploader/e2e/content_uploader_test.ts b/mesop/components/uploader/e2e/content_uploader_test.ts
new file mode 100644
index 00000000..d7b80930
--- /dev/null
+++ b/mesop/components/uploader/e2e/content_uploader_test.ts
@@ -0,0 +1,32 @@
+import {test, expect} from '@playwright/test';
+import path from 'path';
+
+test('test upload file', async ({page}) => {
+ await page.goto('/components/uploader/e2e/content_uploader_app');
+ const fileChooserPromise = page.waitForEvent('filechooser');
+
+ await page.getByText('Upload Image').click();
+ const fileChooser = await fileChooserPromise;
+ await fileChooser.setFiles(path.join(__dirname, 'mesop_robot.jpeg'));
+
+ await expect(page.getByText('File name: mesop_robot.jpeg')).toHaveCount(1);
+ await expect(page.getByText('File size: 30793')).toHaveCount(1);
+ await expect(page.getByText('File type: image/jpeg')).toHaveCount(1);
+ await expect(page.getByText('Upload count: 1')).toHaveCount(1);
+ await expect(
+ page.locator(
+ `//img[@src=""]`,
+ ),
+ ).toHaveCount(1);
+
+ // Check that we can re-upload the same file.
+ // Also check that the icon in the button is being rendered in the composite
+ // uploader component.
+ await page.locator('//*[@role="img" and text()=" upload"]').click();
+ const fileChooser2 = await fileChooserPromise;
+ await fileChooser2.setFiles(path.join(__dirname, 'mesop_robot.jpeg'));
+ await expect(page.getByText('File name: mesop_robot.jpeg')).toHaveCount(1);
+ await expect(page.getByText('File size: 30793')).toHaveCount(1);
+ await expect(page.getByText('File type: image/jpeg')).toHaveCount(1);
+ await expect(page.getByText('Upload count: 2')).toHaveCount(1);
+});
diff --git a/mesop/components/uploader/uploader.ng.html b/mesop/components/uploader/uploader.ng.html
index ae0a5a65..94ea2d01 100644
--- a/mesop/components/uploader/uploader.ng.html
+++ b/mesop/components/uploader/uploader.ng.html
@@ -14,7 +14,7 @@
[style]="getStyle()"
(click)="fileUpload.click()"
>
- {{ config().getLabel() }}
+
} @if(config().getTypeIndex() === 1) {
@@ -26,7 +26,7 @@
[style]="getStyle()"
(click)="fileUpload.click()"
>
- {{ config().getLabel() }}
+
} @if(config().getTypeIndex() === 2) {
@@ -38,7 +38,7 @@
[style]="getStyle()"
(click)="fileUpload.click()"
>
- {{ config().getLabel() }}
+
} @if(config().getTypeIndex() === 3) {
@@ -50,7 +50,19 @@
[style]="getStyle()"
(click)="fileUpload.click()"
>
- {{ config().getLabel() }}
+
+
+
+ } @if(config().getTypeIndex() === 4) {
+
}
diff --git a/mesop/components/uploader/uploader.proto b/mesop/components/uploader/uploader.proto
index 25ad01c3..9ed1d37c 100644
--- a/mesop/components/uploader/uploader.proto
+++ b/mesop/components/uploader/uploader.proto
@@ -15,18 +15,17 @@ message UploadEvent {
}
-// Next ID: 9
+// Next ID: 8
message UploaderType {
- optional string label = 1;
- repeated string accepted_file_type = 2;
- optional string on_upload_event_handler_id = 3;
- optional string color = 4;
- optional bool disable_ripple = 5;
- optional bool disabled = 6;
+ repeated string accepted_file_type = 1;
+ optional string on_upload_event_handler_id = 2;
+ optional string color = 3;
+ optional bool disable_ripple = 4;
+ optional bool disabled = 5;
// Type has two properties:
// |type_index| is used for rendering
// |type| is used for editor value
- optional int32 type_index = 7;
- optional string type = 8;
+ optional int32 type_index = 6;
+ optional string type = 7;
}
diff --git a/mesop/components/uploader/uploader.py b/mesop/components/uploader/uploader.py
index bcc67e23..e058fe1a 100644
--- a/mesop/components/uploader/uploader.py
+++ b/mesop/components/uploader/uploader.py
@@ -4,11 +4,13 @@
import mesop.components.uploader.uploader_pb2 as uploader_pb
from mesop.component_helpers import (
Style,
- insert_component,
+ component,
+ insert_composite_component,
register_event_handler,
register_event_mapper,
register_native_component,
)
+from mesop.components.text.text import text
from mesop.components.uploader.uploaded_file import UploadedFile
from mesop.events import MesopEvent
from mesop.exceptions import MesopDeveloperException
@@ -44,7 +46,7 @@ def map_upload_event(event, key):
register_event_mapper(UploadEvent, map_upload_event)
-@register_native_component
+@component
def uploader(
*,
label: str,
@@ -57,11 +59,51 @@ def uploader(
disabled: bool = False,
style: Style | None = None,
):
+ """Creates an uploader with a simple text Button component.
+
+ Args:
+ label: Uploader button text.
+ accepted_file_types: List of accepted file types. See the [accept parameter](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept).
+ key: The component [key](../components/index.md#component-key).
+ on_upload: File upload event handler.
+ type: Type of button style to use.
+ color: Theme color palette of the button.
+ disable_ripple: Whether the ripple effect is disabled or not.
+ disabled: Whether the button is disabled.
+ style: Style for the component.
"""
- This function creates an uploader.
+ with content_uploader(
+ on_upload=on_upload,
+ accepted_file_types=accepted_file_types,
+ type=type,
+ color=color,
+ disable_ripple=disable_ripple,
+ disabled=disabled,
+ style=style,
+ key=key,
+ ):
+ text(label)
+
+
+@register_native_component
+def content_uploader(
+ *,
+ accepted_file_types: Sequence[str] | None = None,
+ key: str | None = None,
+ on_upload: Callable[[UploadEvent], Any] | None = None,
+ type: Literal["raised", "flat", "stroked", "icon"] | None = None,
+ color: Literal["primary", "accent", "warn"] | None = None,
+ disable_ripple: bool = False,
+ disabled: bool = False,
+ style: Style | None = None,
+):
+ """
+ Creates an uploader component, which is a composite component. Typically, you would
+ use a text or icon component as a child.
+
+ Intended for advanced use cases.
Args:
- label: Upload button label.
accepted_file_types: List of accepted file types. See the [accept parameter](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept).
key: The component [key](../components/index.md#component-key).
on_upload: File upload event handler.
@@ -71,11 +113,10 @@ def uploader(
disabled: Whether the button is disabled.
style: Style for the component.
"""
- insert_component(
+ return insert_composite_component(
key=key,
- type_name="uploader",
+ type_name="content_uploader",
proto=uploader_pb.UploaderType(
- label=label,
accepted_file_type=accepted_file_types or [],
on_upload_event_handler_id=register_event_handler(
on_upload, event=UploadEvent
@@ -93,7 +134,7 @@ def uploader(
def _get_type_index(
- type: Literal["raised", "flat", "stroked"] | None,
+ type: Literal["raised", "flat", "stroked", "icon"] | None,
) -> int:
if type is None:
return 0
@@ -103,4 +144,6 @@ def _get_type_index(
return 2
if type == "stroked":
return 3
+ if type == "icon":
+ return 4
raise Exception("Unexpected type: " + type)
diff --git a/mesop/web/src/component_renderer/type_to_component.ts b/mesop/web/src/component_renderer/type_to_component.ts
index 34569ebe..16044ef4 100644
--- a/mesop/web/src/component_renderer/type_to_component.ts
+++ b/mesop/web/src/component_renderer/type_to_component.ts
@@ -65,7 +65,7 @@ export const typeToComponent = {
'autocomplete': AutocompleteComponent,
'link': LinkComponent,
'html': HtmlComponent,
- 'uploader': UploaderComponent,
+ 'content_uploader': UploaderComponent,
'embed': EmbedComponent,
'table': TableComponent,
'sidenav': SidenavComponent,