Skip to content

Commit

Permalink
ppx: support "custom children" in uppercase components without having…
Browse files Browse the repository at this point in the history
… to wrap in array literal (#823)

* ppx: fix 822

* +changelog
  • Loading branch information
jchavarri authored Nov 24, 2024
1 parent c07e413 commit f95c8c6
Show file tree
Hide file tree
Showing 5 changed files with 63 additions and 17 deletions.
5 changes: 5 additions & 0 deletions CHANGES.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
# Unreleased

* BREAKING, ppx: Allow passing an array of custom children to a component
without having to wrap in array literal ([@jchavarri in #748](https://github.com/reasonml/reason-react/pull/823))

# 0.15.0

* Add `isValidElement` (@r17x in
Expand Down
18 changes: 14 additions & 4 deletions ppx/reason_react_ppx.ml
Original file line number Diff line number Diff line change
Expand Up @@ -458,7 +458,14 @@ let makePropsType ~loc namedTypeList =
};
]

let jsxExprAndChildren ~ident ~loc ~ctxt mapper ~keyProps children =
type component_type = Uppercase | Lowercase

let jsxExprAndChildren ~component_type ~loc ~ctxt mapper ~keyProps children =
let ident =
match component_type with
| Uppercase -> Lident "React"
| Lowercase -> Lident "ReactDOM"
in
let childrenExpr =
Option.map (transformChildrenIfListUpper ~loc ~mapper ~ctxt) children
in
Expand Down Expand Up @@ -492,16 +499,19 @@ let jsxExprAndChildren ~ident ~loc ~ctxt mapper ~keyProps children =
children *)
( Builder.pexp_ident ~loc { loc; txt = Ldot (ident, "jsxs") },
None,
Some (Binding.React.array ~loc children) )
Some
(match component_type with
| Uppercase -> children
| Lowercase -> Binding.React.array ~loc children) )
| None, (label, key) :: _ ->
( Builder.pexp_ident ~loc { loc; txt = Ldot (ident, "jsxKeyed") },
Some (label, key),
None )
| None, [] ->
(Builder.pexp_ident ~loc { loc; txt = Ldot (ident, "jsx") }, None, None)

let reactJsxExprAndChildren = jsxExprAndChildren ~ident:(Lident "React")
let reactDomJsxExprAndChildren = jsxExprAndChildren ~ident:(Lident "ReactDOM")
let reactJsxExprAndChildren = jsxExprAndChildren ~component_type:Uppercase
let reactDomJsxExprAndChildren = jsxExprAndChildren ~component_type:Lowercase

(* Builds an AST node for the entire `external` definition of props *)
let makeExternalDecl fnName loc namedArgListWithKeyAndRef namedTypeList =
Expand Down
4 changes: 2 additions & 2 deletions ppx/test/lower.t/run.t
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@
([@merlin.hide] ReactDOM.domProps)(
~children=
examples
|> List.map(e =>
|> List.map(e => {
let Key = e.path;
ReactDOM.jsxKeyed(
~key=Key,
Expand All @@ -120,7 +120,7 @@
),
(),
);
)
})
|> React.list,
(),
),
Expand Down
5 changes: 1 addition & 4 deletions ppx/test/upper.t/run.t
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,7 @@
let upper_children_single = foo =>
React.jsx(Upper.make, Upper.makeProps(~children=foo, ()));
let upper_children_multiple = (foo, bar) =>
React.jsxs(
Upper.make,
Upper.makeProps(~children=React.array([|foo, bar|]), ()),
);
React.jsxs(Upper.make, Upper.makeProps(~children=[|foo, bar|], ()));
let upper_children =
React.jsx(
Page.make,
Expand Down
48 changes: 41 additions & 7 deletions test/React__test.re
Original file line number Diff line number Diff line change
Expand Up @@ -21,12 +21,17 @@ module DummyComponentThatMapsChildren = {
[@react.component]
let make = (~children, ()) => {
<div>
{children->React.Children.mapWithIndex((element, index) => {
React.cloneElement(
element,
{"key": string_of_int(index), "data-index": index},
)
})}
{children
->React.array
->React.Children.mapWithIndex((element, index) => {
React.cloneElement(
element,
{
"key": string_of_int(index),
"data-index": index,
},
)
})}
</div>;
};
};
Expand Down Expand Up @@ -318,7 +323,10 @@ describe("React", () => {
act(() => {
ReactDOM.Client.render(
root,
render({name: "Joe", imageUrl: "https://foo.png"}),
render({
name: "Joe",
imageUrl: "https://foo.png",
}),
)
});

Expand Down Expand Up @@ -409,4 +417,30 @@ describe("React", () => {

expect(Memo.renders^)->toBe(2);
});

test("Can define components with custom children", () => {
let container = getContainer(container);
let root = ReactDOM.Client.createRoot(container);

module Test = {
type t = {name: string};
[@react.component]
let make = (~children) => {
Array.map(children, c =>
<div name={c.name}> {React.string(c.name)} </div>
)
->React.array;
};
};

act(() => {
ReactDOM.Client.render(
root,
<Test> {Test.name: "foo"} {name: "bar"} </Test>,
)
});

expect(container->DOM.findBySelector("div[name='foo']")->Option.isSome)
->toBe(true);
});
});

0 comments on commit f95c8c6

Please sign in to comment.