Skip to content

Commit 8d6b326

Browse files
Fix multi-child fragment (#852)
* fix multi-child fragment * revert ocamlformat change * refactor: share more code * +changelog --------- Co-authored-by: Antonio Nuno Monteiro <[email protected]>
1 parent b60bd36 commit 8d6b326

File tree

4 files changed

+41
-16
lines changed

4 files changed

+41
-16
lines changed

CHANGES.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,8 @@
33
* Convert `ReasonReactErrorBoundary` to Reason instead of `%raw` JS. This has
44
the benefit of skipping a hardcoded `require('react')` call (@anmonteiro in
55
[#839](https://github.com/reasonml/reason-react/pull/839))
6-
6+
* Fix: Remove "unique `key` prop" warnings from multi-child fragment elements
7+
(@jchavarri in https://github.com/reasonml/reason-react/pull/852)
78

89
# 0.14.1
910

ppx/reason_react_ppx.ml

+35-11
Original file line numberDiff line numberDiff line change
@@ -72,19 +72,25 @@ module Binding = struct
7272
( { loc; txt = Ldot (Lident "React", "componentLike") },
7373
[ props; return ] )
7474

75-
let jsxFragment ~loc ~attrs children =
75+
let makeJsxFragment api ~loc ~attrs children =
7676
let fragment =
7777
Builder.pexp_ident ~loc
7878
{ loc; txt = Ldot (Lident "React", "jsxFragment") }
7979
in
8080
Builder.pexp_apply ~loc ~attrs
81-
(Builder.pexp_ident ~loc { loc; txt = Ldot (Lident "React", "jsx") })
81+
(Builder.pexp_ident ~loc { loc; txt = Ldot (Lident "React", api) })
8282
[
8383
(nolabel, fragment);
8484
( nolabel,
8585
ReactDOM.domProps ~applyLoc:loc ~loc
8686
[ (labelled "children", children); (nolabel, Builder.unit) ] );
8787
]
88+
89+
let jsxFragment ~loc ~attrs children =
90+
makeJsxFragment "jsx" ~loc ~attrs children
91+
92+
let jsxsFragment ~loc ~attrs children =
93+
makeJsxFragment "jsxs" ~loc ~attrs children
8894
end
8995
end
9096

@@ -1393,13 +1399,10 @@ let jsxMapper =
13931399
transformJsxCall ~ctxt parentExpLoc self callExpression
13941400
callArguments nonJSXAttributes)
13951401
(* is it a list with jsx attribute? Reason <>foo</> desugars to
1396-
[@JSX][foo]*)
1402+
[@JSX][foo]
1403+
This will match either <> </> or <> foo </> *)
13971404
| {
1398-
pexp_desc =
1399-
( Pexp_construct
1400-
( { txt = Lident "::"; loc },
1401-
Some { pexp_desc = Pexp_tuple _; _ } )
1402-
| Pexp_construct ({ txt = Lident "[]"; loc }, None) );
1405+
pexp_desc = Pexp_construct ({ txt = Lident ("[]" | "::"); loc }, lst);
14031406
pexp_attributes;
14041407
_;
14051408
} as listItems -> (
@@ -1411,13 +1414,34 @@ let jsxMapper =
14111414
match (jsxAttribute, nonJSXAttributes) with
14121415
(* no JSX attribute *)
14131416
| [], _ -> super#expression ctxt expr
1414-
| _, nonJSXAttributes ->
1417+
| _, nonJSXAttributes -> (
14151418
let childrenExpr =
14161419
transformChildrenIfList ~loc ~ctxt ~mapper:self listItems
14171420
in
14181421
(* throw away the [@JSX] attribute and keep the others, if any *)
1419-
Binding.React.jsxFragment ~loc ~attrs:nonJSXAttributes
1420-
(Binding.React.array ~loc childrenExpr))
1422+
match lst with
1423+
| None
1424+
| Some
1425+
{
1426+
pexp_desc =
1427+
Pexp_tuple
1428+
[
1429+
_;
1430+
{
1431+
pexp_desc =
1432+
Pexp_construct ({ txt = Lident "[]"; _ }, None);
1433+
_;
1434+
};
1435+
];
1436+
_;
1437+
} ->
1438+
Binding.React.jsxFragment ~loc ~attrs:nonJSXAttributes
1439+
(Binding.React.array ~loc childrenExpr)
1440+
| Some { pexp_desc = Pexp_tuple (_ :: _); _ } ->
1441+
(* Fragment with two or more children: <> foo bar </> *)
1442+
Binding.React.jsxsFragment ~loc ~attrs:nonJSXAttributes
1443+
(Binding.React.array ~loc childrenExpr)
1444+
| _ -> assert false))
14211445
(* Delegate to the default mapper, a deep identity traversal *)
14221446
| e -> super#expression ctxt e
14231447
[@@raises Invalid_argument]

ppx/test/component.t/run.t

+1-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ We need to output ML syntax here, otherwise refmt could not parse it.
2727
""[@@mel.obj ]
2828
let make =
2929
((fun ?(name= "") ->
30-
React.jsx React.jsxFragment
30+
React.jsxs React.jsxFragment
3131
(((ReactDOM.domProps)[@merlin.hide ])
3232
~children:(React.array
3333
[|(ReactDOM.jsx "div"

ppx/test/fragment.t/run.t

+3-3
Original file line numberDiff line numberDiff line change
@@ -11,21 +11,21 @@
1111
([@merlin.hide] ReactDOM.domProps)(~children=React.array([|bar|]), ()),
1212
);
1313
let poly_children_fragment = (foo, bar) =>
14-
React.jsx(
14+
React.jsxs(
1515
React.jsxFragment,
1616
([@merlin.hide] ReactDOM.domProps)(
1717
~children=React.array([|foo, bar|]),
1818
(),
1919
),
2020
);
2121
let nested_fragment = (foo, bar, baz) =>
22-
React.jsx(
22+
React.jsxs(
2323
React.jsxFragment,
2424
([@merlin.hide] ReactDOM.domProps)(
2525
~children=
2626
React.array([|
2727
foo,
28-
React.jsx(
28+
React.jsxs(
2929
React.jsxFragment,
3030
([@merlin.hide] ReactDOM.domProps)(
3131
~children=React.array([|bar, baz|]),

0 commit comments

Comments
 (0)