From 55f37b4576668cf029a03d9aac322ba49b5aeed0 Mon Sep 17 00:00:00 2001 From: Dan Abramov Date: Tue, 4 Oct 2016 19:33:09 +0100 Subject: [PATCH] Update the homepage with ES6 (#7868) * Update the homepage with ES6 * Avoid array spread and stale state --- Rakefile | 4 +-- _js/examples/hello.js | 11 +++---- _js/examples/markdown.js | 29 ++++++++++-------- _js/examples/timer.js | 37 +++++++++++++---------- _js/examples/todo.js | 63 +++++++++++++++++++++++++--------------- _js/live_editor.js | 20 +++++++++---- _layouts/default.html | 2 +- 7 files changed, 101 insertions(+), 65 deletions(-) diff --git a/Rakefile b/Rakefile index d50dc5b1..7a9872b7 100644 --- a/Rakefile +++ b/Rakefile @@ -6,8 +6,8 @@ require('open-uri') desc "download babel-browser" task :fetch_remotes do IO.copy_stream( - open('https://cdnjs.cloudflare.com/ajax/libs/babel-core/5.8.34/browser.min.js'), - 'js/babel-browser.min.js' + open('https://unpkg.com/babel-standalone@6.15.0/babel.min.js'), + 'js/babel.min.js' ) end diff --git a/_js/examples/hello.js b/_js/examples/hello.js index bab86329..16f7f1a4 100644 --- a/_js/examples/hello.js +++ b/_js/examples/hello.js @@ -1,12 +1,13 @@ +var name = Math.random() > 0.5 ? 'Jane' : 'John'; var HELLO_COMPONENT = ` -var HelloMessage = React.createClass({ - render: function() { +class HelloMessage extends React.Component { + render() { return
Hello {this.props.name}
; } -}); +} -ReactDOM.render(, mountNode); -`; +ReactDOM.render(, mountNode); +`.trim(); ReactDOM.render( , diff --git a/_js/examples/markdown.js b/_js/examples/markdown.js index 8ec68ac6..696074cd 100644 --- a/_js/examples/markdown.js +++ b/_js/examples/markdown.js @@ -1,16 +1,21 @@ var MARKDOWN_COMPONENT = ` -var MarkdownEditor = React.createClass({ - getInitialState: function() { - return {value: 'Type some *markdown* here!'}; - }, - handleChange: function() { +class MarkdownEditor extends React.Component { + constructor(props) { + super(props); + this.handleChange = this.handleChange.bind(this); + this.state = {value: 'Type some *markdown* here!'}; + } + + handleChange() { this.setState({value: this.refs.textarea.value}); - }, - rawMarkup: function() { + } + + getRawMarkup() { var md = new Remarkable(); return { __html: md.render(this.state.value) }; - }, - render: function() { + } + + render() { return (

Input

@@ -21,15 +26,15 @@ var MarkdownEditor = React.createClass({

Output

); } -}); +} ReactDOM.render(, mountNode); -`; +`.trim(); ReactDOM.render( , diff --git a/_js/examples/timer.js b/_js/examples/timer.js index af9ebbd8..3d8cfc4c 100644 --- a/_js/examples/timer.js +++ b/_js/examples/timer.js @@ -1,26 +1,33 @@ var TIMER_COMPONENT = ` -var Timer = React.createClass({ - getInitialState: function() { - return {secondsElapsed: 0}; - }, - tick: function() { - this.setState({secondsElapsed: this.state.secondsElapsed + 1}); - }, - componentDidMount: function() { - this.interval = setInterval(this.tick, 1000); - }, - componentWillUnmount: function() { +class Timer extends React.Component { + constructor(props) { + super(props); + this.state = {secondsElapsed: 0}; + } + + tick() { + this.setState((prevState) => ({ + secondsElapsed: prevState.secondsElapsed + 1 + })); + } + + componentDidMount() { + this.interval = setInterval(() => this.tick(), 1000); + } + + componentWillUnmount() { clearInterval(this.interval); - }, - render: function() { + } + + render() { return (
Seconds Elapsed: {this.state.secondsElapsed}
); } -}); +} ReactDOM.render(, mountNode); -`; +`.trim(); ReactDOM.render( , diff --git a/_js/examples/todo.js b/_js/examples/todo.js index cc7c979c..eaac631d 100644 --- a/_js/examples/todo.js +++ b/_js/examples/todo.js @@ -1,41 +1,56 @@ var TODO_COMPONENT = ` -var TodoList = React.createClass({ - render: function() { - var createItem = function(item) { - return
  • {item.text}
  • ; - }; - return
      {this.props.items.map(createItem)}
    ; +class TodoApp extends React.Component { + constructor(props) { + super(props); + this.handleChange = this.handleChange.bind(this); + this.handleSubmit = this.handleSubmit.bind(this); + this.state = {items: [], text: ''}; } -}); -var TodoApp = React.createClass({ - getInitialState: function() { - return {items: [], text: ''}; - }, - onChange: function(e) { - this.setState({text: e.target.value}); - }, - handleSubmit: function(e) { - e.preventDefault(); - var nextItems = this.state.items.concat([{text: this.state.text, id: Date.now()}]); - var nextText = ''; - this.setState({items: nextItems, text: nextText}); - }, - render: function() { + + render() { return (

    TODO

    - +
    ); } -}); + + handleChange(e) { + this.setState({text: e.target.value}); + } + + handleSubmit(e) { + e.preventDefault(); + var newItem = { + text: this.state.text, + id: Date.now() + }; + this.setState((prevState) => ({ + items: prevState.items.concat(newItem), + text: '' + })); + } +} + +class TodoList extends React.Component { + render() { + return ( +
      + {this.props.items.map(item => ( +
    • {item.text}
    • + ))} +
    + ); + } +} ReactDOM.render(, mountNode); -`; +`.trim(); ReactDOM.render( , diff --git a/_js/live_editor.js b/_js/live_editor.js index 27b4c81a..134cae5b 100644 --- a/_js/live_editor.js +++ b/_js/live_editor.js @@ -90,8 +90,14 @@ var ReactPlayground = React.createClass({ getDefaultProps: function() { return { - transformer: function(code) { - return babel.transform(code).code; + transformer: function(code, options) { + var presets = ['react']; + if (!options || !options.skipES2015Transform) { + presets.push('es2015'); + } + return Babel.transform(code, { + presets + }).code; }, editorTabTitle: 'Live JSX Editor', showCompiledJSTab: true, @@ -115,15 +121,15 @@ var ReactPlayground = React.createClass({ this.setState({mode: mode}); }, - compileCode: function() { - return this.props.transformer(this.state.code); + compileCode: function(options) { + return this.props.transformer(this.state.code, options); }, render: function() { var isJS = this.state.mode === this.MODES.JS; var compiledCode = ''; try { - compiledCode = this.compileCode(); + compiledCode = this.compileCode({skipES2015Transform: true}); } catch (err) {} var JSContent = @@ -201,13 +207,15 @@ var ReactPlayground = React.createClass({ } catch (e) { } try { - var compiledCode = this.compileCode(); + var compiledCode; if (this.props.renderCode) { + compiledCode = this.compileCode({skipES2015Transform: true}); ReactDOM.render( , mountNode ); } else { + compiledCode = this.compileCode({skipES2015Transform: false}); eval(compiledCode); } } catch (err) { diff --git a/_layouts/default.html b/_layouts/default.html index d42b8b15..a3a44a53 100644 --- a/_layouts/default.html +++ b/_layouts/default.html @@ -36,7 +36,7 @@ - +