Skip to content

Commit 178fc05

Browse files
authored
Add support for ComponentType children in vdom_to_html (#1257)
1 parent a54ce4e commit 178fc05

File tree

3 files changed

+52
-30
lines changed

3 files changed

+52
-30
lines changed

docs/source/about/changelog.rst

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ Unreleased
2121
- :pull:`1239` - Script elements no longer support behaving like effects. They now strictly behave like plain HTML script elements.
2222
- :pull:`1255` - The ``reactpy.html`` module has been modified to allow for auto-creation of any HTML nodes. For example, you can create a ``<data-table>`` element by calling ``html.data_table()``.
2323
- :pull:`1256` - Change ``set_state`` comparison method to check equality with ``==`` more consistently.
24+
- :pull:`1257` - Add support for rendering ``@component`` children within ``vdom_to_html``.
2425

2526
**Removed**
2627

src/reactpy/utils.py

+33-25
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,13 @@
33
import re
44
from collections.abc import Iterable
55
from itertools import chain
6-
from typing import Any, Callable, Generic, TypeVar, cast
6+
from typing import Any, Callable, Generic, TypeVar, Union, cast
77

88
from lxml import etree
99
from lxml.html import fromstring, tostring
1010

11-
from reactpy.core.types import VdomDict
12-
from reactpy.core.vdom import vdom
11+
from reactpy.core.types import ComponentType, VdomDict
12+
from reactpy.core.vdom import vdom as make_vdom
1313

1414
_RefValue = TypeVar("_RefValue")
1515
_ModelTransform = Callable[[VdomDict], Any]
@@ -144,7 +144,7 @@ def _etree_to_vdom(
144144
children = _generate_vdom_children(node, transforms)
145145

146146
# Convert the lxml node to a VDOM dict
147-
el = vdom(node.tag, dict(node.items()), *children)
147+
el = make_vdom(node.tag, dict(node.items()), *children)
148148

149149
# Perform any necessary mutations on the VDOM attributes to meet VDOM spec
150150
_mutate_vdom(el)
@@ -160,7 +160,7 @@ def _add_vdom_to_etree(parent: etree._Element, vdom: VdomDict | dict[str, Any])
160160
try:
161161
tag = vdom["tagName"]
162162
except KeyError as e:
163-
msg = f"Expected a VDOM dict, not {vdom}"
163+
msg = f"Expected a VDOM dict, not {type(vdom)}"
164164
raise TypeError(msg) from e
165165
else:
166166
vdom = cast(VdomDict, vdom)
@@ -174,29 +174,29 @@ def _add_vdom_to_etree(parent: etree._Element, vdom: VdomDict | dict[str, Any])
174174
element = parent
175175

176176
for c in vdom.get("children", []):
177+
if hasattr(c, "render"):
178+
c = _component_to_vdom(cast(ComponentType, c))
177179
if isinstance(c, dict):
178180
_add_vdom_to_etree(element, c)
181+
182+
# LXML handles string children by storing them under `text` and `tail`
183+
# attributes of Element objects. The `text` attribute, if present, effectively
184+
# becomes that element's first child. Then the `tail` attribute, if present,
185+
# becomes a sibling that follows that element. For example, consider the
186+
# following HTML:
187+
188+
# <p><a>hello</a>world</p>
189+
190+
# In this code sample, "hello" is the `text` attribute of the `<a>` element
191+
# and "world" is the `tail` attribute of that same `<a>` element. It's for
192+
# this reason that, depending on whether the element being constructed has
193+
# non-string a child element, we need to assign a `text` vs `tail` attribute
194+
# to that element or the last non-string child respectively.
195+
elif len(element):
196+
last_child = element[-1]
197+
last_child.tail = f"{last_child.tail or ''}{c}"
179198
else:
180-
"""
181-
LXML handles string children by storing them under `text` and `tail`
182-
attributes of Element objects. The `text` attribute, if present, effectively
183-
becomes that element's first child. Then the `tail` attribute, if present,
184-
becomes a sibling that follows that element. For example, consider the
185-
following HTML:
186-
187-
<p><a>hello</a>world</p>
188-
189-
In this code sample, "hello" is the `text` attribute of the `<a>` element
190-
and "world" is the `tail` attribute of that same `<a>` element. It's for
191-
this reason that, depending on whether the element being constructed has
192-
non-string a child element, we need to assign a `text` vs `tail` attribute
193-
to that element or the last non-string child respectively.
194-
"""
195-
if len(element):
196-
last_child = element[-1]
197-
last_child.tail = f"{last_child.tail or ''}{c}"
198-
else:
199-
element.text = f"{element.text or ''}{c}"
199+
element.text = f"{element.text or ''}{c}"
200200

201201

202202
def _mutate_vdom(vdom: VdomDict) -> None:
@@ -249,6 +249,14 @@ def _generate_vdom_children(
249249
)
250250

251251

252+
def _component_to_vdom(component: ComponentType) -> VdomDict | str | None:
253+
"""Convert a component to a VDOM dictionary"""
254+
result = component.render()
255+
if hasattr(result, "render"):
256+
result = _component_to_vdom(cast(ComponentType, result))
257+
return cast(Union[VdomDict, str, None], result)
258+
259+
252260
def del_html_head_body_transform(vdom: VdomDict) -> VdomDict:
253261
"""Transform intended for use with `html_to_vdom`.
254262

tests/test_utils.py

+18-5
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
import pytest
44

55
import reactpy
6-
from reactpy import html
6+
from reactpy import component, html
77
from reactpy.utils import (
88
HTMLParseError,
99
del_html_head_body_transform,
@@ -193,6 +193,21 @@ def test_del_html_body_transform():
193193
SOME_OBJECT = object()
194194

195195

196+
@component
197+
def example_parent():
198+
return example_middle()
199+
200+
201+
@component
202+
def example_middle():
203+
return html.div({"id": "sample", "style": {"padding": "15px"}}, example_child())
204+
205+
206+
@component
207+
def example_child():
208+
return html.h1("Sample Application")
209+
210+
196211
@pytest.mark.parametrize(
197212
"vdom_in, html_out",
198213
[
@@ -254,10 +269,8 @@ def test_del_html_body_transform():
254269
'<div data-something="1" data-something-else="2" dataisnotdashed="3"></div>',
255270
),
256271
(
257-
html.div(
258-
{"dataSomething": 1, "dataSomethingElse": 2, "dataisnotdashed": 3}
259-
),
260-
'<div data-something="1" data-something-else="2" dataisnotdashed="3"></div>',
272+
html.div(example_parent()),
273+
'<div><div id="sample" style="padding:15px"><h1>Sample Application</h1></div></div>',
261274
),
262275
],
263276
)

0 commit comments

Comments
 (0)