-
Notifications
You must be signed in to change notification settings - Fork 464
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Jsx ast #7286
Jsx ast #7286
Conversation
|
||
(* | ||
* jsx-fragment ::= | ||
* | <> </> | ||
* | <> jsx-children </> | ||
*) | ||
and parse_jsx_fragment p = | ||
and parse_jsx_fragment start_pos p = |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I prefer ranges to be accurate. The location should start at the opening <
token.
@@ -1000,7 +1000,7 @@ Path Objects.Rec. | |||
|
|||
Complete src/Completion.res 120:7 | |||
posCursor:[120:7] posNoWhite:[120:6] Found expr:[119:11->123:1] | |||
posCursor:[120:7] posNoWhite:[120:6] Found expr:[120:5->122:5] | |||
posCursor:[120:7] posNoWhite:[120:6] Found expr:[120:5->123:0] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Changes like this are to be expected as the range of a fragment spans from <>
till end </>
.
posCursor:[9:56] posNoWhite:[9:55] Found expr:[9:13->9:66] | ||
JSX <SectionHeader:[9:13->9:26] > _children:9:26 | ||
posCursor:[9:56] posNoWhite:[9:55] Found expr:__ghost__[9:10->9:67] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This ghost expression is a part of the AstHelper.make_list_expression
result.
It is no longer present in the new AST, but it also didn't serve any purpose.
I believe it is okay that this test is slightly different.
In the end the result didn't change.
compiler/ml/parsetree.ml
Outdated
(* [%id] *) | ||
(* . *) | ||
(* represents <> foo </> , the entire range is stored in the expression , we keep track of >, children and </ *) | ||
| Pexp_jsx_fragment of |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The reason the store the >
and </
token is this edge case:
https://rescript-lang.org/try?version=v12.0.0-alpha.8&module=esmodule&code=DYUwLgBA+hC8ECgCQAeCB6AVBAzmATgIYB2A5iBJuhAHzJIDeASiIQMZgB0e+AlmQAoARAFsQACyEBKAL7IU1LBEIiIASQh9S4yFVpA
If we ever want to restore comments I suppose we need the proper anchors.
Map child expressions Initial mapping of Pexp_jsx_fragment to 0 Correct location in mapping Update analysis for jsx_fragment Remove unused code Print something for ml print Commit invalid test results for reference Try improve printing Correct fragment range, try and print comments Indent jsx Process comments from children inside fragment Attach comments to fragment tags Fix comment Improve comment formatting Print single element on same line Update comment WIP: Debug More debugging Works Fix some jsx printing Fix the test Clean up Update tests with location changes
Thank you @shulhi for fixing all the remaining formatter problem of the fragment node! |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Left some comments.
Just checking: is the conversion complete in that the representation of fragments is completely moved over to the new representation?
compiler/ml/ast_mapper_to0.ml
Outdated
@@ -407,6 +407,18 @@ module E = struct | |||
| Pexp_open (ovf, lid, e) -> | |||
open_ ~loc ~attrs ovf (map_loc sub lid) (sub.expr sub e) | |||
| Pexp_extension x -> extension ~loc ~attrs (sub.extension sub x) | |||
| Pexp_jsx_fragment (o, xs, c) -> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There should be a corresponding change in ast_mapper_from0
that inverts the conversion.
Also, some extension of the test cases in tests/tools_tests/ppx/TestPpx.res
to check that back and forth conversion works fine.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Alright, I think the new tests from #7318 prove this now.
Yes, that is the idea. The parser only produces jsx_fragment and not the old representation. |
tests/tools_tests/ppx/TestPpx.res
Outdated
@@ -61,3 +61,7 @@ let eq2 = 3 === 3 | |||
|
|||
let test = async () => 12 | |||
let f = async () => (await test()) + 1 | |||
|
|||
module Fragments = { | |||
let f1 = <> </> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should be <></>
, no? (without the space).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sounds reasonable, although I do believe this is the current behavior. (Playground)
@cristianoc, anything else you can think of for |
This could be ready to go. Is the absence of changes in |
Would you add a changelog too? Assuming this change will go ahead, and the rest of the JSX investigated in a separate PR. |
Yes, I understand. Today, I made some necessary changes to ensure the tests remained consistent.
I would recommend investigating this in the same PR. There isn't much benefit to having just the fragments part. To avoid the risk of not completing a second PR, I would suggest keeping it as one. That being said, I would like to achieve a sense of completion for the fragments before taking on more changes. |
Sure continuing jsx sounds good. |
Alright, I think I’ve figured out all the printing details! This is ready for review!
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks great! That was some work!!
Left some minor comments on the implementation.
The first order of business is that compilation is not affected, and that seems fine.
Second, the changes in printing. They look good in tests. Perhaps worth trying on an existing large project to see what the differences are in the field. @cknitt any candidates?
| Pexp_apply {funct = {pexp_desc = Pexp_ident compName}; args} | ||
when Res_parsetree_viewer.is_jsx_expression expr -> | ||
| Pexp_jsx_element | ||
( Jsx_unary_element |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
thoughts about pros/cons of unifying the jsx ast nodes into one?
In this case, perhaps one could streamline the code a bit more, but don't know about the rest
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe combining these is beneficial, although it may not be a game changer. If you don't need to handle JSX, you can avoid a single branch in your match. However, if you do need to handle them, you'll need to address all of them.
@@ -383,7 +382,7 @@ let jsx_child_expr expr = | |||
( Pexp_ident _ | Pexp_constant _ | Pexp_field _ | Pexp_construct _ | |||
| Pexp_variant _ | Pexp_array _ | Pexp_pack _ | Pexp_record _ | |||
| Pexp_extension _ | Pexp_letmodule _ | Pexp_letexception _ | |||
| Pexp_open _ | Pexp_sequence _ | Pexp_let _ ); | |||
| Pexp_open _ | Pexp_sequence _ | Pexp_let _ | Pexp_jsx_element _ ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
double checking: only 1 of the 3 jsx ast nodes here is correct?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This captures all three nodes, so what exactly do you mean here?
Resolved opens 1 Stdlib | ||
Path Comp.make | ||
{"contents": {"kind": "markdown", "value": "```rescript\nint\n```"}} | ||
{"contents": {"kind": "markdown", "value": "```rescript\nComp.props<int>\n```\n\n---\n\n```\n \n```\n```rescript\ntype Comp.props<'age> = {age: 'age}\n```\nGo to: [Type definition](command:rescript-vscode.go_to_location?%5B%22Jsx2.res%22%2C157%2C2%5D)\n"}} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
was this incorrect before?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not really, but after these changes there was type information for this hover, where previously there wasn't. I suspect this is the case because of more accurate ranges in the node now.
Forgot to ask: this seems complete (minus some leftover commented code in a couple of files). Anything known to be missing / less confident about? |
So like I said printer bugs with comments might be a thing. I believe I undid, #7269 And, I'm not sure if anyone is doing something with ppx with this, if so, might be interesting to hear if that still works for them. |
Alright, I found a solution to the missing blank lines, similar to how #7269 introduced it. This is ready for some testing on real world projects. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks great, absolutely fantastic job! 🎉 ⭐ 👏
jsx_container_element_children = | ||
JSXChildrenSpreading _ | JSXChildrenItems (_ :: _); | ||
}) -> | ||
(* This is a weird edge case where there is no closing tag but there are children *) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
<div>{React.string("hello")
Is that an example of such a case?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yup!
let try_map_jsx_prop (sub : mapper) (lbl : Asttypes.Noloc.arg_label) | ||
(e : expression) : Parsetree.jsx_prop option = | ||
match (lbl, e) with | ||
| Asttypes.Noloc.Labelled "_spreadProps", expr -> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should the constant _spreadProps
be defined somewhere central perhaps, instead of being inlined? Does it matter?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It is used 4 times, so seems like worth doing, do you have any place in mind?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If there's a place where logic around JSX prop spreading is located, that's probably a good place.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⭐
compiler/ml/parsetree.ml
Outdated
} | ||
|
||
and jsx_unary_element = { | ||
(* jsx_unary_element_opening: Lexing.position; *) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove these leftover comments...?
* | lident | ||
* | ?lident | ||
*) | ||
| JSXPropPunning of (* optional *) bool * (* name *) string loc |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not that important, but sometimes I wonder if we should just default to inline records instead of tuples.
and jsx_closing_container_tag = { | ||
(* </ *) | ||
jsx_closing_container_tag_start: Lexing.position; | ||
(* name *) | ||
jsx_closing_container_tag_name: Longident.t loc; | ||
(* > *) | ||
jsx_closing_container_tag_end: Lexing.position; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great comments!
compiler/ml/pprintast.ml
Outdated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Very cool to get this in the actual printer now.
@@ -12,7 +12,6 @@ | |||
|
|||
module InstallerDownload = | |||
struct | |||
let make [arity:1]() = ((div ~children:[] ())[@res.braces ][@JSX ]) | |||
[@@react.component ] | |||
let make [arity:1]() = ((<div />)[@res.braces ])[@@react.component ] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
😍
Very nice work! I can do some testing on real-world projects later today or tomorrow. |
{ | ||
jsx_container_element_tag_name_start = compName; | ||
jsx_container_element_props = props; | ||
} ) -> | ||
inJsxContext := true; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm curious why do we need inJsxContext
flag since we are already in JSX branch? Maybe we can document this decision somewhere in the code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is used in CompletionBackEnd to autocomplete ->React.string
when dotting into a string. CPField
and CPPipe
do have a comment for it.
This is awesome! 🎉 Just tested it against a large project (that also uses a PPX) with no issues whatsoever. Everything compiled fine and the JS output was unchanged compared to the alpha.11 release. Reformatting the sources led to minor changes regarding fragments and comments that I all consider improvements: Before: let renderWithParens = el => <>
{t(nbsp ++ "(")}
el
{t(")")}
</> After: let renderWithParens = el =>
<>
{t(nbsp ++ "(")}
el
{t(")")}
</> Before: <input
disabled
className="hidden"
id=/* accept=".*" */
"fileUploadButton"
type_="file"
multiple=false
onChange
onClick After: <input
disabled
className="hidden"
id=/* accept=".*" */ "fileUploadButton"
type_="file"
multiple=false
onChange
onClick |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ready to go!
This is a part of #7283.
I'm introducing
Pexp_jsx_fragment
to represent fragment syntax<></>
.I found it insightful to just try it out and see what code changes are necessary.
In short a fragment is now parsed as:
after this change it becomes:
I'll add some comment to relevant changes.