diff --git a/py/examples/graphics_background.py b/py/examples/graphics_background.py new file mode 100644 index 0000000000..7727b6cb23 --- /dev/null +++ b/py/examples/graphics_background.py @@ -0,0 +1,19 @@ +# Graphics / Background Image +# Set a background image behind your #graphics. +# Original example: https://docs.python.org/3/library/turtle.html +# --- +from h2o_wave import site, ui, graphics as g + +t = g.turtle().f(100).r(90).pd() +for _ in range(36): + t.f(200).l(170) +spirograph = t.pu(1).path(stroke='red', fill='yellow') + +page = site['/demo'] +page['example'] = ui.graphics_card( + box='1 1 3 4', view_box='0 0 220 220', width='100%', height='100%', + scene=g.scene(foo=spirograph), + image_path='https://images.pexels.com/photos/1269968/pexels-photo-1269968.jpeg?auto=compress', +) + +page.save() diff --git a/py/examples/tour.conf b/py/examples/tour.conf index 8e52c5b338..4b2dbaa640 100644 --- a/py/examples/tour.conf +++ b/py/examples/tour.conf @@ -254,4 +254,5 @@ graphics_clock.py graphics_path.py graphics_turtle.py graphics_hilbert.py +graphics_background.py glider_gun.py diff --git a/py/h2o_lightwave/h2o_lightwave/types.py b/py/h2o_lightwave/h2o_lightwave/types.py index 855b47747f..39d71e6321 100644 --- a/py/h2o_lightwave/h2o_lightwave/types.py +++ b/py/h2o_lightwave/h2o_lightwave/types.py @@ -8984,12 +8984,18 @@ def __init__( scene: Optional[PackedData] = None, width: Optional[str] = None, height: Optional[str] = None, + image: Optional[str] = None, + image_path: Optional[str] = None, + image_type: Optional[str] = None, commands: Optional[List[Command]] = None, ): _guard_scalar('GraphicsCard.box', box, (str,), False, False, False) _guard_scalar('GraphicsCard.view_box', view_box, (str,), False, False, False) _guard_scalar('GraphicsCard.width', width, (str,), False, True, False) _guard_scalar('GraphicsCard.height', height, (str,), False, True, False) + _guard_scalar('GraphicsCard.image', image, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_path', image_path, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_type', image_type, (str,), False, True, False) _guard_vector('GraphicsCard.commands', commands, (Command,), False, True, False) self.box = box """A string indicating how to place this component on the page.""" @@ -9003,6 +9009,12 @@ def __init__( """The displayed width of the rectangular viewport. (Not the width of its coordinate system.)""" self.height = height """The displayed height of the rectangular viewport. (Not the height of its coordinate system.)""" + self.image = image + """Background image data, base64-encoded.""" + self.image_path = image_path + """The path or URL or data URL of the background image, e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`.""" + self.image_type = image_type + """The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. Required only if `image` is set.""" self.commands = commands """Contextual menu commands for this component.""" @@ -9012,6 +9024,9 @@ def dump(self) -> Dict: _guard_scalar('GraphicsCard.view_box', self.view_box, (str,), False, False, False) _guard_scalar('GraphicsCard.width', self.width, (str,), False, True, False) _guard_scalar('GraphicsCard.height', self.height, (str,), False, True, False) + _guard_scalar('GraphicsCard.image', self.image, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_path', self.image_path, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_type', self.image_type, (str,), False, True, False) _guard_vector('GraphicsCard.commands', self.commands, (Command,), False, True, False) return _dump( view='graphics', @@ -9021,6 +9036,9 @@ def dump(self) -> Dict: scene=self.scene, width=self.width, height=self.height, + image=self.image, + image_path=self.image_path, + image_type=self.image_type, commands=None if self.commands is None else [__e.dump() for __e in self.commands], ) @@ -9037,6 +9055,12 @@ def load(__d: Dict) -> 'GraphicsCard': _guard_scalar('GraphicsCard.width', __d_width, (str,), False, True, False) __d_height: Any = __d.get('height') _guard_scalar('GraphicsCard.height', __d_height, (str,), False, True, False) + __d_image: Any = __d.get('image') + _guard_scalar('GraphicsCard.image', __d_image, (str,), False, True, False) + __d_image_path: Any = __d.get('image_path') + _guard_scalar('GraphicsCard.image_path', __d_image_path, (str,), False, True, False) + __d_image_type: Any = __d.get('image_type') + _guard_scalar('GraphicsCard.image_type', __d_image_type, (str,), False, True, False) __d_commands: Any = __d.get('commands') _guard_vector('GraphicsCard.commands', __d_commands, (dict,), False, True, False) box: str = __d_box @@ -9045,6 +9069,9 @@ def load(__d: Dict) -> 'GraphicsCard': scene: Optional[PackedData] = __d_scene width: Optional[str] = __d_width height: Optional[str] = __d_height + image: Optional[str] = __d_image + image_path: Optional[str] = __d_image_path + image_type: Optional[str] = __d_image_type commands: Optional[List[Command]] = None if __d_commands is None else [Command.load(__e) for __e in __d_commands] return GraphicsCard( box, @@ -9053,6 +9080,9 @@ def load(__d: Dict) -> 'GraphicsCard': scene, width, height, + image, + image_path, + image_type, commands, ) diff --git a/py/h2o_lightwave/h2o_lightwave/ui.py b/py/h2o_lightwave/h2o_lightwave/ui.py index 46327b4518..6eda9f5844 100644 --- a/py/h2o_lightwave/h2o_lightwave/ui.py +++ b/py/h2o_lightwave/h2o_lightwave/ui.py @@ -3121,6 +3121,9 @@ def graphics_card( scene: Optional[PackedData] = None, width: Optional[str] = None, height: Optional[str] = None, + image: Optional[str] = None, + image_path: Optional[str] = None, + image_type: Optional[str] = None, commands: Optional[List[Command]] = None, ) -> GraphicsCard: """Create a card for displaying vector graphics. @@ -3132,6 +3135,9 @@ def graphics_card( scene: Foreground layer for rendering dynamic SVG elements. width: The displayed width of the rectangular viewport. (Not the width of its coordinate system.) height: The displayed height of the rectangular viewport. (Not the height of its coordinate system.) + image: Background image data, base64-encoded. + image_path: The path or URL or data URL of the background image, e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`. + image_type: The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. Required only if `image` is set. commands: Contextual menu commands for this component. Returns: A `h2o_wave.types.GraphicsCard` instance. @@ -3143,6 +3149,9 @@ def graphics_card( scene, width, height, + image, + image_path, + image_type, commands, ) diff --git a/py/h2o_wave/h2o_wave/types.py b/py/h2o_wave/h2o_wave/types.py index 855b47747f..39d71e6321 100644 --- a/py/h2o_wave/h2o_wave/types.py +++ b/py/h2o_wave/h2o_wave/types.py @@ -8984,12 +8984,18 @@ def __init__( scene: Optional[PackedData] = None, width: Optional[str] = None, height: Optional[str] = None, + image: Optional[str] = None, + image_path: Optional[str] = None, + image_type: Optional[str] = None, commands: Optional[List[Command]] = None, ): _guard_scalar('GraphicsCard.box', box, (str,), False, False, False) _guard_scalar('GraphicsCard.view_box', view_box, (str,), False, False, False) _guard_scalar('GraphicsCard.width', width, (str,), False, True, False) _guard_scalar('GraphicsCard.height', height, (str,), False, True, False) + _guard_scalar('GraphicsCard.image', image, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_path', image_path, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_type', image_type, (str,), False, True, False) _guard_vector('GraphicsCard.commands', commands, (Command,), False, True, False) self.box = box """A string indicating how to place this component on the page.""" @@ -9003,6 +9009,12 @@ def __init__( """The displayed width of the rectangular viewport. (Not the width of its coordinate system.)""" self.height = height """The displayed height of the rectangular viewport. (Not the height of its coordinate system.)""" + self.image = image + """Background image data, base64-encoded.""" + self.image_path = image_path + """The path or URL or data URL of the background image, e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`.""" + self.image_type = image_type + """The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. Required only if `image` is set.""" self.commands = commands """Contextual menu commands for this component.""" @@ -9012,6 +9024,9 @@ def dump(self) -> Dict: _guard_scalar('GraphicsCard.view_box', self.view_box, (str,), False, False, False) _guard_scalar('GraphicsCard.width', self.width, (str,), False, True, False) _guard_scalar('GraphicsCard.height', self.height, (str,), False, True, False) + _guard_scalar('GraphicsCard.image', self.image, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_path', self.image_path, (str,), False, True, False) + _guard_scalar('GraphicsCard.image_type', self.image_type, (str,), False, True, False) _guard_vector('GraphicsCard.commands', self.commands, (Command,), False, True, False) return _dump( view='graphics', @@ -9021,6 +9036,9 @@ def dump(self) -> Dict: scene=self.scene, width=self.width, height=self.height, + image=self.image, + image_path=self.image_path, + image_type=self.image_type, commands=None if self.commands is None else [__e.dump() for __e in self.commands], ) @@ -9037,6 +9055,12 @@ def load(__d: Dict) -> 'GraphicsCard': _guard_scalar('GraphicsCard.width', __d_width, (str,), False, True, False) __d_height: Any = __d.get('height') _guard_scalar('GraphicsCard.height', __d_height, (str,), False, True, False) + __d_image: Any = __d.get('image') + _guard_scalar('GraphicsCard.image', __d_image, (str,), False, True, False) + __d_image_path: Any = __d.get('image_path') + _guard_scalar('GraphicsCard.image_path', __d_image_path, (str,), False, True, False) + __d_image_type: Any = __d.get('image_type') + _guard_scalar('GraphicsCard.image_type', __d_image_type, (str,), False, True, False) __d_commands: Any = __d.get('commands') _guard_vector('GraphicsCard.commands', __d_commands, (dict,), False, True, False) box: str = __d_box @@ -9045,6 +9069,9 @@ def load(__d: Dict) -> 'GraphicsCard': scene: Optional[PackedData] = __d_scene width: Optional[str] = __d_width height: Optional[str] = __d_height + image: Optional[str] = __d_image + image_path: Optional[str] = __d_image_path + image_type: Optional[str] = __d_image_type commands: Optional[List[Command]] = None if __d_commands is None else [Command.load(__e) for __e in __d_commands] return GraphicsCard( box, @@ -9053,6 +9080,9 @@ def load(__d: Dict) -> 'GraphicsCard': scene, width, height, + image, + image_path, + image_type, commands, ) diff --git a/py/h2o_wave/h2o_wave/ui.py b/py/h2o_wave/h2o_wave/ui.py index 46327b4518..6eda9f5844 100644 --- a/py/h2o_wave/h2o_wave/ui.py +++ b/py/h2o_wave/h2o_wave/ui.py @@ -3121,6 +3121,9 @@ def graphics_card( scene: Optional[PackedData] = None, width: Optional[str] = None, height: Optional[str] = None, + image: Optional[str] = None, + image_path: Optional[str] = None, + image_type: Optional[str] = None, commands: Optional[List[Command]] = None, ) -> GraphicsCard: """Create a card for displaying vector graphics. @@ -3132,6 +3135,9 @@ def graphics_card( scene: Foreground layer for rendering dynamic SVG elements. width: The displayed width of the rectangular viewport. (Not the width of its coordinate system.) height: The displayed height of the rectangular viewport. (Not the height of its coordinate system.) + image: Background image data, base64-encoded. + image_path: The path or URL or data URL of the background image, e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`. + image_type: The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. Required only if `image` is set. commands: Contextual menu commands for this component. Returns: A `h2o_wave.types.GraphicsCard` instance. @@ -3143,6 +3149,9 @@ def graphics_card( scene, width, height, + image, + image_path, + image_type, commands, ) diff --git a/r/R/ui.R b/r/R/ui.R index 5d7cae0ddd..450d48637f 100644 --- a/r/R/ui.R +++ b/r/R/ui.R @@ -3639,6 +3639,11 @@ ui_frame_card <- function( #' (Not the width of its coordinate system.) #' @param height The displayed height of the rectangular viewport. #' (Not the height of its coordinate system.) +#' @param image Background image data, base64-encoded. +#' @param image_path The path or URL or data URL of the background image, +#' e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`. +#' @param image_type The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. +#' Required only if `image` is set. #' @param commands Contextual menu commands for this component. #' @return A GraphicsCard instance. #' @export @@ -3649,6 +3654,9 @@ ui_graphics_card <- function( scene = NULL, width = NULL, height = NULL, + image = NULL, + image_path = NULL, + image_type = NULL, commands = NULL) { .guard_scalar("box", "character", box) .guard_scalar("view_box", "character", view_box) @@ -3656,6 +3664,9 @@ ui_graphics_card <- function( # TODO Validate scene: Data .guard_scalar("width", "character", width) .guard_scalar("height", "character", height) + .guard_scalar("image", "character", image) + .guard_scalar("image_path", "character", image_path) + .guard_scalar("image_type", "character", image_type) .guard_vector("commands", "WaveCommand", commands) .o <- list( box=box, @@ -3664,6 +3675,9 @@ ui_graphics_card <- function( scene=scene, width=width, height=height, + image=image, + image_path=image_path, + image_type=image_type, commands=commands, view='graphics') class(.o) <- append(class(.o), c(.wave_obj, "WaveGraphicsCard")) diff --git a/tools/intellij-plugin/src/main/resources/templates/wave-components.xml b/tools/intellij-plugin/src/main/resources/templates/wave-components.xml index a0c89ee6af..5b93918c0a 100644 --- a/tools/intellij-plugin/src/main/resources/templates/wave-components.xml +++ b/tools/intellij-plugin/src/main/resources/templates/wave-components.xml @@ -1473,13 +1473,16 @@ - + + + + diff --git a/tools/vscode-extension/component-snippets.json b/tools/vscode-extension/component-snippets.json index af67cf3644..f370cd430d 100644 --- a/tools/vscode-extension/component-snippets.json +++ b/tools/vscode-extension/component-snippets.json @@ -1234,7 +1234,7 @@ "Wave Full GraphicsCard": { "prefix": "w_full_graphics_card", "body": [ - "ui.graphics_card(box='$1', view_box='$2', stage=${3:None}, scene=${4:None}, width='$5', height='$6', commands=[\n\t\t$7\t\t\n])$0" + "ui.graphics_card(box='$1', view_box='$2', stage=${3:None}, scene=${4:None}, width='$5', height='$6', image='$7', image_path='$8', image_type='$9', commands=[\n\t\t$10\t\t\n])$0" ], "description": "Create a full Wave GraphicsCard." }, diff --git a/ui/src/graphics.tsx b/ui/src/graphics.tsx index 9ec8b3a8f0..6771e1475f 100644 --- a/ui/src/graphics.tsx +++ b/ui/src/graphics.tsx @@ -43,6 +43,18 @@ interface State { * (Not the height of its coordinate system.) */ height?: S + /** Background image data, base64-encoded. */ + image?: S + /** + * The path or URL or data URL of the background image, + * e.g. `/foo.png` or `http://example.com/foo.png` or `data:image/png;base64,???`. + */ + image_path?: S + /** + * The background image MIME subtype. One of `apng`, `bmp`, `gif`, `x-icon`, `jpeg`, `png`, `webp`. + * Required only if `image` is set. + */ + image_type?: S } const @@ -169,14 +181,22 @@ export const }, render = () => { const - { view_box, width, height, stage, scene } = state, + { view_box, width, height, stage, scene, image_type, image, image_path } = state, stageEls = stage ? unpack(stage).map(renderEl) : [], sceneEls = scene ? unpack(scene).map(({ d, o }, i) => renderEl({ ...(d ? JSON.parse(d) : {}), ...(o ? JSON.parse(o) : {}), - }, i)) : [] + }, i)) : [], + backgroundImageSrc = image_path + ? 'url(' + image_path + ')' + : image ? `url(data:image/${image_type};base64,${image})` + : undefined + return ( - + {stageEls} {sceneEls} diff --git a/website/docs/examples/assets/graphics-background.png b/website/docs/examples/assets/graphics-background.png new file mode 100644 index 0000000000..b581b3c7db Binary files /dev/null and b/website/docs/examples/assets/graphics-background.png differ diff --git a/website/sidebars.js b/website/sidebars.js index 7168b91ac7..01c06ba07d 100644 --- a/website/sidebars.js +++ b/website/sidebars.js @@ -60,6 +60,7 @@ module.exports = { "widgets/content/tabs", "widgets/content/markdown", "widgets/content/image", + "widgets/content/graphics", "widgets/content/article", "widgets/content/tall_article_preview", "widgets/content/wide_article_preview", diff --git a/website/widgets/content/graphics.md b/website/widgets/content/graphics.md new file mode 100644 index 0000000000..6bcf5b0d59 --- /dev/null +++ b/website/widgets/content/graphics.md @@ -0,0 +1,37 @@ +--- +title: Graphics +keywords: + - graphics +custom_edit_url: null +--- + +To render vector graphics (SVG) in your app, use the graphics card. + +```py +from h2o_wave import graphics + +red_square = graphics.p().m(25, 25).h(50).v(50).h(-50).z().path(fill='red') + +q.page['example'] = ui.graphics_card( + box='1 1 2 3', view_box='0 0 100 100', width='100%', height='100%', + scene=graphics.scene(foo=red_square), +) +``` + +See the [Graphics](https://wave.h2o.ai/docs/graphics) section for a detailed explanation of graphics support. Full API at [ui.graphics_card](http://wave.h2o.ai/docs/api/ui#graphics_card). + +## Background image + +Set a background image on the graphics card, either by specifying the image's path or by providing base64-encoded image data. + +```py +from h2o_wave import graphics + +red_square = graphics.p().m(25, 25).h(50).v(50).h(-50).z().path(fill='red') + +q.page['example'] = ui.graphics_card( + box='1 1 2 3', view_box='0 0 100 100', width='100%', height='100%', + scene=graphics.scene(foo=red_square), + image_path='https://images.pexels.com/photos/1269968/pexels-photo-1269968.jpeg?auto=compress', +) +``` diff --git a/website/widgets/overview.md b/website/widgets/overview.md index a8f5b6c32e..fc85d2975c 100644 --- a/website/widgets/overview.md +++ b/website/widgets/overview.md @@ -35,6 +35,7 @@ native Wave components and use custom HTML only as a last resort solution. - [tabs](/docs/widgets/content/tabs) - [markdown](/docs/widgets/content/markdown) - [image](/docs/widgets/content/image) +- [graphics](/docs/widgets/content/graphics) - [article](/docs/widgets/content/article) - [tall_article_preview](/docs/widgets/content/tall_article_preview) - [wide_article_preview](/docs/widgets/content/wide_article_preview)