JSX is an XML-like syntax extension to ECMAScript without any defined semantics. It's NOT intended to be implemented by engines or browsers. It's NOT a proposal to incorporate JSX into the ECMAScript spec itself. It's intended to be used by various preprocessors (transpilers) to transform these tokens into standard ECMAScript.

var dropdown = <Dropdown > A dropdown list <Menu > <MenuItem > Do Something< / MenuItem > <MenuItem > Do Something Fun ! < / MenuItem > <MenuItem > Do Something Else< / MenuItem > < / Menu > < / Dropdown > ; render ( dropdown ) ;

The purpose of this specification is to define a concise and familiar syntax for defining tree structures with attributes. A generic but well defined syntax enables a community of independent parsers and syntax highlighters to conform to a single specification.

Embedding a new syntax in an existing language is a risky venture. Other syntax implementors or the existing language may introduce another incompatible syntax extension.

Through a stand-alone specification, we make it easier for implementors of other syntax extensions to consider JSX when designing their own syntax. This will hopefully allow various new syntax extensions to co-exist.

It is our intention to claim minimal syntactic real estate while keeping the syntax concise and familiar. That way we leave the door open for other extensions.

This specification does not attempt to comply with any XML or HTML specification. JSX is designed as an ECMAScript feature and the similarity to XML is only for familiarity.

JSX extends the PrimaryExpression in the ECMAScript 6th Edition (ECMA-262) grammar:

PrimaryExpression :

JSXElement

JSXFragment

Elements

JSXElement :

JSXSelfClosingElement

JSXOpeningElement JSXChildren opt JSXClosingElement<br /> (names of JSXOpeningElement and JSXClosingElement should match)

JSXSelfClosingElement :

< JSXElementName JSXAttributes opt / >

JSXOpeningElement :

< JSXElementName JSXAttributes opt >

JSXClosingElement :

< / JSXElementName >

JSXFragment :

< > JSXChildren opt < / >

JSXElementName :

JSXIdentifier

JSXNamespacedName

JSXMemberExpression

JSXIdentifier :

IdentifierStart

JSXIdentifier IdentifierPart

JSXIdentifier NO WHITESPACE OR COMMENT -

JSXNamespacedName :

JSXIdentifier : JSXIdentifier

JSXMemberExpression :

JSXIdentifier . JSXIdentifier

JSXIdentifier JSXMemberExpression . JSXIdentifier

Attributes

JSXAttributes :

JSXSpreadAttribute JSXAttributes opt

JSXAttribute JSXAttributes opt

JSXSpreadAttribute :

{ ... AssignmentExpression }

JSXAttribute :

JSXAttributeName JSXAttributeInitializer opt

JSXAttributeName :

JSXIdentifier

JSXNamespacedName

JSXAttributeInitializer :

= JSXAttributeValue

JSXAttributeValue :

" JSXDoubleStringCharacters opt "

JSXDoubleStringCharacters ' JSXSingleStringCharacters opt '

JSXSingleStringCharacters { AssignmentExpression }

AssignmentExpression JSXElement

JSXFragment

JSXDoubleStringCharacters :

JSXDoubleStringCharacter JSXDoubleStringCharacters opt

JSXDoubleStringCharacter :

SourceCharacter but not "

JSXSingleStringCharacters :

JSXSingleStringCharacter JSXSingleStringCharacters opt

JSXSingleStringCharacter :

SourceCharacter but not '

Children

JSXChildren :

JSXChild JSXChildren opt

JSXChild :

JSXText

JSXElement

{ JSXChildExpression opt }

JSXText :

JSXTextCharacter JSXText opt

JSXTextCharacter :

SourceCharacter but not one of { , < , > or }

JSXChildExpression :

AssignmentExpression

... AssignmentExpression

Whitespace and Comments

JSX uses the same punctuators and braces as ECMAScript. WhiteSpace, LineTerminators and Comments are generally allowed between any punctuators.

acorn-jsx: A fork of acorn.

esprima-fb: A fork of esprima.

jsx-reader: A sweet.js macro.

These are a set of transpilers that all conform to the JSX syntax but use different semantics on the output:

React JSX: Create ReactElements using JSX.

jsx-transform: Configurable implementation of JSX decoupled from React.

Babel: An ES2015 and beyond to ES of now transpiler with JSX support.

NOTE: A conforming transpiler may choose to use a subset of the JSX syntax.

Why not Template Literals? #

ECMAScript 6th Edition (ECMA-262) introduces template literals which are intended to be used for embedding DSL in ECMAScript. Why not just use that instead of inventing a syntax that's not part of ECMAScript?

Template literals work well for long embedded DSLs. Unfortunately the syntax noise is substantial when you exit in and out of embedded arbitrary ECMAScript expressions with identifiers in scope.

var box = jsx` <$ { Box } > $ { shouldShowAnswer ( user ) ? jsx`<$ { Answer } value = $ { false } > no< / $ { Answer } > ` : jsx` <$ { Box . Comment } > Text Content < / $ { Box . Comment } > ` } < / $ { Box } > ` ;

It would be possible to use template literals as a syntactic entry point and change the semantics inside the template literal to allow embedded scripts that can be evaluated in scope:

var box = jsx` <Box > { shouldShowAnswer ( user ) ? <Answer value = { false } > no< / Answer > : <Box . Comment > Text Content < / Box . Comment > } < / Box > ` ;

However, this would lead to further divergence. Tooling that is built around the assumptions imposed by template literals wouldn't work. It would undermine the meaning of template literals. It would be necessary to define how JSX behaves within the rest of the ECMAScript grammar within the template literal anyway.

Therefore it's better to introduce JSX as an entirely new type of PrimaryExpression:

var box = <Box > { shouldShowAnswer ( user ) ? <Answer value = { false } > no< / Answer > : <Box . Comment > Text Content < / Box . Comment > } < / Box > ;

Another alternative would be to use object initializers (similar to JXON). Unfortunately, the balanced braces do not give great syntactic hints for where an element starts and ends in large trees. Balanced named tags is a critical syntactic feature of the XML-style notation.

The JSX syntax is similar to the E4X Specification (ECMA-357). E4X is a deprecated specification with deep reaching semantic meaning. JSX partially overlaps with a tiny subset of the E4X syntax. However, JSX has no relation to the E4X specification.

Copyright (c) 2014 - present, Facebook, Inc. All rights reserved.

This work is licensed under a Creative Commons Attribution 4.0 International License.