diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cb7ec6a7c..41383dfe4d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,6 +12,9 @@ # 12.0.0-alpha.12 (Unreleased) +#### :house: Internal +- Better representation of JSX in AST . https://github.com/rescript-lang/rescript/pull/7286 + # 12.0.0-alpha.11 #### :bug: Bug fix diff --git a/analysis/src/CompletionFrontEnd.ml b/analysis/src/CompletionFrontEnd.ml index 79044efad8..e2d4a51f46 100644 --- a/analysis/src/CompletionFrontEnd.ml +++ b/analysis/src/CompletionFrontEnd.ml @@ -1233,8 +1233,6 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor then ValueOrField else Value); })) - | Pexp_construct ({txt = Lident ("::" | "()")}, _) -> - (* Ignore list expressions, used in JSX, unit, and more *) () | Pexp_construct (lid, eOpt) -> ( let lidPath = flattenLidCheckDot lid in if debug then @@ -1325,10 +1323,29 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor inJsx = !inJsxContext; })) | None -> ()) - | Pexp_apply {funct = {pexp_desc = Pexp_ident compName}; args} - when Res_parsetree_viewer.is_jsx_expression expr -> + | Pexp_jsx_element + ( Jsx_unary_element + { + jsx_unary_element_tag_name = compName; + jsx_unary_element_props = props; + } + | Jsx_container_element + { + jsx_container_element_tag_name_start = compName; + jsx_container_element_props = props; + } ) -> inJsxContext := true; - let jsxProps = CompletionJsx.extractJsxProps ~compName ~args in + let children = + match expr.pexp_desc with + | Pexp_jsx_element + (Jsx_container_element + {jsx_container_element_children = children}) -> + children + | _ -> JSXChildrenItems [] + in + let jsxProps = + CompletionJsx.extractJsxProps ~compName ~props ~children + in let compNamePath = flattenLidCheckDot ~jsx:true compName in if debug then Printf.printf "JSX <%s:%s %s> _children:%s\n" @@ -1345,10 +1362,21 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor | None -> "None" | Some childrenPosStart -> Pos.toString childrenPosStart); let jsxCompletable = - CompletionJsx.findJsxPropsCompletable ~jsxProps - ~endPos:(Loc.end_ expr.pexp_loc) ~posBeforeCursor - ~posAfterCompName:(Loc.end_ compName.loc) - ~firstCharBeforeCursorNoWhite ~charAtCursor + match expr.pexp_desc with + | Pexp_jsx_element + (Jsx_container_element + { + jsx_container_element_closing_tag = None; + jsx_container_element_children = + JSXChildrenSpreading _ | JSXChildrenItems (_ :: _); + }) -> + (* This is a weird edge case where there is no closing tag but there are children *) + None + | _ -> + CompletionJsx.findJsxPropsCompletable ~jsxProps + ~endPos:(Loc.end_ expr.pexp_loc) ~posBeforeCursor + ~posAfterCompName:(Loc.end_ compName.loc) + ~firstCharBeforeCursorNoWhite ~charAtCursor in if jsxCompletable <> None then setResultOpt jsxCompletable else if compName.loc |> Loc.hasPos ~pos:posBeforeCursor then diff --git a/analysis/src/CompletionJsx.ml b/analysis/src/CompletionJsx.ml index 5014a665c8..0c49ae0086 100644 --- a/analysis/src/CompletionJsx.ml +++ b/analysis/src/CompletionJsx.ml @@ -455,40 +455,39 @@ let findJsxPropsCompletable ~jsxProps ~endPos ~posBeforeCursor in loop jsxProps.props -let extractJsxProps ~(compName : Longident.t Location.loc) ~args = - let thisCaseShouldNotHappen = - { - compName = Location.mknoloc (Longident.Lident ""); - props = []; - childrenStart = None; - } +let extractJsxProps ~(compName : Longident.t Location.loc) ~props ~children = + let open Parsetree in + let childrenStart = + match children with + | JSXChildrenItems [] -> None + | JSXChildrenSpreading child | JSXChildrenItems (child :: _) -> + if child.pexp_loc.loc_ghost then None else Some (Loc.start child.pexp_loc) in - let rec processProps ~acc args = - match args with - | (Asttypes.Labelled {txt = "children"}, {Parsetree.pexp_loc}) :: _ -> - { - compName; - props = List.rev acc; - childrenStart = - (if pexp_loc.loc_ghost then None else Some (Loc.start pexp_loc)); - } - | ( (Labelled {txt = s; loc} | Optional {txt = s; loc}), - (eProp : Parsetree.expression) ) - :: rest -> ( - let namedArgLoc = if loc = Location.none then None else Some loc in - match namedArgLoc with - | Some loc -> - processProps - ~acc: - ({ - name = s; - posStart = Loc.start loc; - posEnd = Loc.end_ loc; - exp = eProp; - } - :: acc) - rest - | None -> processProps ~acc rest) - | _ -> thisCaseShouldNotHappen + let props = + props + |> List.map (function + | JSXPropPunning (_, name) -> + { + name = name.txt; + posStart = Loc.start name.loc; + posEnd = Loc.end_ name.loc; + exp = + Ast_helper.Exp.ident ~loc:name.loc + {txt = Longident.Lident name.txt; loc = name.loc}; + } + | JSXPropValue (name, _, value) -> + { + name = name.txt; + posStart = Loc.start name.loc; + posEnd = Loc.end_ name.loc; + exp = value; + } + | JSXPropSpreading (loc, expr) -> + { + name = "_spreadProps"; + posStart = Loc.start loc; + posEnd = Loc.end_ loc; + exp = expr; + }) in - args |> processProps ~acc:[] + {compName; props; childrenStart} diff --git a/analysis/src/SemanticTokens.ml b/analysis/src/SemanticTokens.ml index cbd435a5c4..10219f1b54 100644 --- a/analysis/src/SemanticTokens.ml +++ b/analysis/src/SemanticTokens.ml @@ -120,9 +120,7 @@ let emitLongident ?(backwards = false) ?(jsx = false) let rec flatten acc lid = match lid with | Longident.Lident txt -> txt :: acc - | Ldot (lid, txt) -> - let acc = if jsx && txt = "createElement" then acc else txt :: acc in - flatten acc lid + | Ldot (lid, txt) -> flatten (txt :: acc) lid | _ -> acc in let rec loop pos segments = @@ -247,8 +245,8 @@ let command ~debug ~emitter ~path = ~posEnd:(Some (Loc.end_ loc)) ~lid ~debug; Ast_iterator.default_iterator.expr iterator e - | Pexp_apply {funct = {pexp_desc = Pexp_ident lident; pexp_loc}; args} - when Res_parsetree_viewer.is_jsx_expression e -> + | Pexp_jsx_element (Jsx_unary_element {jsx_unary_element_tag_name = lident}) + -> (* Angled brackets: - These are handled in the grammar: <> > /> @@ -258,46 +256,56 @@ let command ~debug ~emitter ~path = - handled like other Longitent.t, except lowercase id is marked Token.JsxLowercase *) emitter (* -->
{React.string("moo")}
+ // c1 +{React.string("moo")}
+ // c2 + // c3 +{React.string("moo")}
+ + // c4 + +{React.string("moo")}
+{React.string("moo")}
+ // c1 +{React.string("moo")}
+ // c2 + // c3 +{React.string("moo")}
+ + // c4 + +{React.string("moo")}
+ > + +let arrow_with_fragment = el => <> + {t(nbsp ++ "(")} + el + {t(")")} +> + +let arrow_with_container_tag = el => +{React.string("moo")}
+ // c1 +{React.string("moo")}
+ // c2 + // c3 +{React.string("moo")}
+ // c4 + +{React.string("moo")}
+{React.string("moo")}
+ // c1 +{React.string("moo")}
+ // c2 + // c3 +{React.string("moo")}
+ // c4 + +{React.string("moo")}
+ > + +let arrow_with_fragment = el => <> + {t(nbsp ++ "(")} + el + {t(")")} +> + +let arrow_with_container_tag = el =>