React in the form we usually see doesn't really look like it can make components dynamically. Most people end up using switch
with case
blocks to "choose" the type of component that will be rendered when there are multiple possibilities, which I'd say is an anti-pattern.
For example:
import React from 'react'; // required in order to use JSX | |
import Type1 from './Type1'; // load one type of component | |
import Type2 from './Type2'; // load a second type of component | |
// a component that will render one or more components | |
// based on a switch statement | |
export const SwitchedComponent = (props) => { | |
let Output; // save the rendered JSX to return | |
// check the type of the component given in props | |
switch ( props.type ) { | |
// render Type1 with props | |
case 'Type1': | |
Output = (<Type1 { ...props } />); | |
break; | |
// render Type2 with props | |
case 'Type2': | |
Output = (<Type2 { ...props } />); | |
break; | |
// unknown type ... output null to not render | |
default: | |
Output = (null); // to return nothing, use null | |
break; | |
} | |
return Output; // return the output created in switch/case | |
}; | |
export default SwitchedComponent; |
We don't want to have to have that switch statement ... as the number of components we might output grows, it's going to start getting really ugly and hard to maintain. The page JSX in Depth in the React documentation holds the key to understanding how to create components dynamically instead, in the section marked "Choosing the Type at Runtime". That page doesn't come up high in searches, though, and it doesn't do a very good job of explaining the technique.
I like to call this technique the Capitalized reference technique.
Here's an explanation of how to use the technique:
- Import the components we might use
- Add references to the components to an object literal
- Create a reference to the dynamic component type we want by:
- Creating a new variable (reference) with a first letter that is Capitalized
- Using the component type as the key, get the corresponding value from the object literal
- Assign the value from the literal, a reference to the component, to the Capitalized reference
- Include the dynamic component using the Capitalized reference from step 3.1 in our JSX.
The fundamental 'magic' here is that when JSX sees the Capitalized reference, it dereferences back to whatever component the reference is pointed at. Yay references!
In the simplest form:
import React from 'react'; // required in order to use JSX | |
import Type1 from './Type1'; // load one type of component | |
import Type2 from './Type2'; // load a second type of component | |
// make references to the components by type | |
const Components = { | |
Type1, | |
Type2 | |
}; | |
// a component that will render one or more components | |
// that it doesn't explicitly include in it's JSX | |
export const CapitalizedReferenceComponent = (props) => { | |
// make a reference using a Capitalized variable name | |
// to the component you need to render | |
// where props.type is one of 'Type1' or 'Type2' | |
let Component = Components[ props.type ]; | |
// use the reference to the component with the | |
// Capitalized variable name to render it | |
return (<Component { ...props } />); | |
}; | |
export default CapitalizedReferenceComponent; |
Requiring us to import any component we might want to render and add it to an object literal isn't very maintainable. We want to keep the list of possible components to render outside of this dynamic component renderer. To do that, we'll send a components dictionary in as part of props
. After we externalize the set of possible components to render, we'll achieve good maintainability.
import React from 'react'; // required in order to use JSX | |
import Type1 from './Type1'; // load one type of component | |
import Type2 from './Type2'; // load a second type of component | |
// make references to the components by type | |
const Components = { | |
Type1, | |
Type2 | |
}; | |
// a component that will render one or more components | |
// that it doesn't explicitly include in it's JSX | |
export const CapitalizedReferenceComponent = (props) => { | |
// make a reference using a Capitalized variable name | |
// to the component you need to render | |
// where props.type is one of 'Type1' or 'Type2' | |
let Component = Components[ props.type ]; | |
// use the reference to the component with the | |
// Capitalized variable name to render it | |
return (<Component { ...props } />); | |
}; | |
export default CapitalizedReferenceComponent; |
We probably aren't rendering just one component this way, but rendering a wrapper around any number of dynamically-defined children. To work on a collection of dynamic component instances, we need to:
- Iterate over the collection in JSX
- For each item:
- Reassign the value we get from
Components[ component.type ]
to the Capitalized reference we've created - Use that Capitalized reference in our JSX in order to render the correct component type
- Reassign the value we get from
import React from 'react'; // required to use JSX | |
export const CapitalizedReferenceComponentCollectionExternals = (props) => { | |
// get references to all possible components | |
// that this component might render, | |
// and the collection of dynamic components we need to render, | |
// using destructuring | |
const { components: Components, collection } = props; | |
// A Capitalized reference to reuse | |
let Component; | |
// A reference to the component's props to reuse | |
let componentProps; | |
// A function which returns component.props if it exists, | |
// and otherwise returns props | |
const defaultMapPropsToComponent = function ( | |
{ | |
component = {}, | |
props = {} | |
} = {} | |
) { | |
return component.props || props; | |
}; | |
// render the component collection | |
return( | |
<div>{ | |
collection.map( | |
( component ) => { | |
// Reference the proper component | |
Component = Components[ component.type ]; | |
// Get the props you want to use for this component instance | |
// here we are assuming that you can specify a mapping function | |
// on the component definition, in props, or use the default | |
componentProps = ( | |
component.mapPropsToComponent || | |
props.mapPropsToComponent || | |
defaultMapPropsToComponent | |
)( component, props ); | |
return (<Component { ...componentProps } />); | |
} | |
) | |
}</div> | |
); | |
}; | |
export default CapitalizedReferenceComponentCollectionExternals; |
The method for iterating over a collection in JSX is also covered on the page JSX in Depth, this time in a section titled "javascript expressions as children".
Now that we can easily render components based on a configuration, we've achieved a basic kind of polymorphism -- the type of a component can be changed by changing a value in props. Doing it based off of a collection proves that we can use any kind of technique that walks an object tree to render a hierarchy in order to achieve composition. The arrangement and functioning of an application can be driven based on the collection.
A complete sample project and demo can be found here: react-dynamic-component-demo.