From 4d285a70a7f6be806c1dbcaf9d26d29cb8f25c44 Mon Sep 17 00:00:00 2001 From: MICHAEL JACKSON Date: Mon, 27 Nov 2017 13:25:23 -0800 Subject: [PATCH] Add doc about Render Props --- content/docs/nav.yml | 4 +- content/docs/render-props.md | 224 +++++++++++++++++++++++++++++++++++ 2 files changed, 227 insertions(+), 1 deletion(-) create mode 100644 content/docs/render-props.md diff --git a/content/docs/nav.yml b/content/docs/nav.yml index ecb84207..e147339e 100644 --- a/content/docs/nav.yml +++ b/content/docs/nav.yml @@ -56,6 +56,8 @@ title: Web Components - id: higher-order-components title: Higher-Order Components + - id: render-props + title: Render Props - id: integrating-with-other-libraries title: Integrating with Other Libraries - id: accessibility @@ -110,4 +112,4 @@ - id: faq-structure title: File Structure - id: faq-internals - title: Virtual DOM and Internals \ No newline at end of file + title: Virtual DOM and Internals diff --git a/content/docs/render-props.md b/content/docs/render-props.md new file mode 100644 index 00000000..6991daac --- /dev/null +++ b/content/docs/render-props.md @@ -0,0 +1,224 @@ +--- +id: render-props +title: Render Props +permalink: docs/render-props.html +--- + +The term ["render prop"](https://cdb.reacttraining.com/use-a-render-prop-50de598f11ce) refers to a simple technique for sharing code between React components using a prop whose value is a function. + +## Use Render Props for Cross-Cutting Concerns + +Components are the primary unit of code reuse in React, but it's not always obvious how to share the state or behavior that one component encapsulates to other components that need that same state. + +For example, the following component tracks the mouse position in a web app: + +```js +class MouseTracker extends React.Component { + constructor(props) { + super(props); + this.handleMouseMove = this.handleMouseMove.bind(this); + this.state = { x: 0, y: 0 }; + } + + handleMouseMove(event) { + this.setState({ + x: event.clientX, + y: event.clientY + }); + } + + render() { + return ( +
+

Move the mouse around!

+

The current mouse position is ({this.state.x}, {this.state.y})

+
+ ); + } +} +``` + +As the cursor moves around the screen, the component displays its (x, y) coordinates in a `

`. + +Now the question is: How can we reuse this behavior in another component? In other words, if another component needs to know about the cursor position, can we encapsulate that behavior so that we can easily share it with that component? + +Since components are the basic unit of code reuse in React, let's try refactoring the code a bit to use a `` component that encapsulates the behavior we need to reuse elsewhere. + +```js +// The component encapsulates the behavior we need... +class Mouse extends React.Component { + constructor(props) { + super(props); + this.handleMouseMove = this.handleMouseMove.bind(this); + this.state = { x: 0, y: 0 }; + } + + handleMouseMove(event) { + this.setState({ + x: event.clientX, + y: event.clientY + }); + } + + render() { + return ( +

+ + {/* ...but how do we render something other than a

? */} +

The current mouse position is ({this.state.x}, {this.state.y})

+
+ ); + } +} + +class MouseTracker extends React.Component { + render() { + return ( +
+

Move the mouse around!

+ +
+ ); + } +} +``` + +Now the `` component encapsulates all behavior associated with listening for `mousemove` events and storing the (x, y) position of the cursor, but it's not yet truly reusable. + +For example, let's say we have a `` component that renders the image of a cat chasing the mouse around the screen. We might use a `` prop to tell the component the coordinates of the mouse so it knows where to position the image on the screen. + +As a first pass, you might try rendering the `` *inside ``'s `render` method*, like this: + +```js +class Cat extends React.Component { + render() { + const mouse = this.props.mouse + return ( + + ); + } +} + +class MouseWithCat extends React.Component { + constructor(props) { + super(props); + this.handleMouseMove = this.handleMouseMove.bind(this); + this.state = { x: 0, y: 0 }; + } + + handleMouseMove(event) { + this.setState({ + x: event.clientX, + y: event.clientY + }); + } + + render() { + return ( +
+ + {/* + We could just swap out the

for a here ... but then + we would need to create a separate + component every time we need to use it, so + isn't really reusable yet. + */} + +

+ ); + } +} + +class MouseTracker extends React.Component { + render() { + return ( +
+

Move the mouse around!

+ +
+ ); + } +} +``` + +This approach will work for our specific use case, but we haven't achieved the objective of truly encapsulating the behavior in a reusable way. Now, every time we want the mouse position for a different use case, we have to create a new component (i.e. essentially another ``) that renders something specifically for that use case. + +Here's where the render prop comes in: Instead of hard-coding a `` inside a `` component, and effectively changing its rendered output, we can provide `` with a function prop that it uses to dynamically determine what to render–a render prop. + +```js +class Cat extends React.Component { + render() { + const mouse = this.props.mouse; + return ( + + ); + } +} + +class Mouse extends React.Component { + constructor(props) { + super(props); + this.handleMouseMove = this.handleMouseMove.bind(this); + this.state = { x: 0, y: 0 }; + } + + handleMouseMove(event) { + this.setState({ + x: event.clientX, + y: event.clientY + }); + } + + render() { + return ( +
+ + {/* + Instead of providing a static representation of what renders, + use the `render` prop to dynamically determine what to render. + */} + {this.props.render(this.state)} +
+ ); + } +} + +class MouseTracker extends React.Component { + render() { + return ( +
+

Move the mouse around!

+ ( + + )}/> +
+ ); + } +} +``` + +Now, instead of effectively cloning the `` component and hard-coding something else in its `render` method to solve for a specific use case, we instead provide a `render` prop that `` can use to dynamically determine what it renders. + +More concretely, **a render prop is a function prop that a component uses to know what to render.** + +This technique makes the behavior that we need to share extremely portable. To get that behavior, render a `` with a `render` prop that tells it what to render with the current (x, y) of the cursor. + +One interesting thing to note about render props is that you can implement most [higher-order components](/react/docs/higher-order-components.html) (HOC) using a regular component with a render prop. For example, if you would prefer to have a `withMouse` HOC instead of a `` component, you could easily create one using a regular `` with a render prop: + +```js +// If you really want a HOC for some reason, you can easily +// create one using a regular component with a render prop! +function withMouse(Component) { + return class extends React.Component { + render() { + return ( + ( + + )}/> + ); + } + } +} +``` + +So using a render prop makes it possible to use either pattern.