React Native

Ohana.js

twitter.com/keokilee | facebook.com/georgelee

github.com/keokilee

4/23/2015

Get Started

http://github.com/ohanajs/react-native-workshop

Step 1: Getting set up

Requirements

  • Homebrew
  • Xcode
  • Node (brew install node)
  • Watchman (brew install watchman)
  • React Native CLI (npm install -g react-native-cli)

Start a project

  • react-native init AppWithAMap
  • cd AppWithAMap
  • open AppWithAMap.xcodeproj

Let's Play Around

  • Keep Xcode and the terminal window running at all times
  • Open the AppWithAMap directory in your text editor
  • Edit index.ios.js
  • Reload the page

Add Navigator

  • Include NavigatorIOS
  • Create component with initialRoute
  • Make sure it has a style of "flex: 1"!!!
var AppWithAMap = React.createClass({
  render() {
    return (
      <NavigatorIOS style={styles.navigator} 
        initialRoute={{
          component: Welcome,
          title: "Welcome"
        }}
      />
    )
  }
});

var Welcome = React.createClass({
  ...
});

var styles = StyleSheet.create({
  navigator: {
    flex: 1
  },
  ...
});

Step 2: ListView

Basic App Structure

  • Create a folder called components/map-points
  • Create a file in there called "index.js"
  • Move "Welcome" component and its styles
  • Export the welcome component and include it in index.ios.js

Change to a ListView

  • Import ListView from React Native
  • Use getInitialState and create a data source (rowHasChanged and cloneWithRows)
  • Render the List View with a simple row

Refactor Row

  • Let's create a new component for our row
  • Pass the row as a prop
  • Render the text inside a View
  • Apply some styles (borderBottom, padding)
var MapPoints = React.createClass({
  getInitialState () {
    var dataSource = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});

    return {
      dataSource: dataSource.cloneWithRows(['foo', 'bar', 'baz'])
    }
  },
  render: function() {
    return (
      <ListView
        dataSource={this.state.dataSource}
        renderRow={(row) => <MapRow row={row} />} />
    );
  }
});

var MapRow = React.createClass({
  render() {
    return (
      <View style={styles.row}>
        <Text style={styles.text}>{this.props.row}</Text>
      </View>
    );
  }
});

Wait, We Need Map Points

  • Grab the mock_data.js file from http://github.com/ohanajs/react-native-workshop
  • Include it and require it into our map-points component
  • Use those rows as the data source.
  • Update the MapRow to display the latitude and longitude.
var MapRow = React.createClass({
  render() {
    var latitude = this.props.row.geometry.latitude;
    var longitude = this.props.row.geometry.longitude;
    var text = `Latitude: ${latitude}\nLongitude: ${longitude}`;

    return (
      <View style={styles.row}>
        <Text style={styles.text}>
          {text}
        </Text>
      </View>
    );
  }
});

Step 3: Um, Can I Haz Map Plz?

Create a new component

  • Create a new folder in components called "map-display"
  • Create an index.js file in "map-display"
  • Create a MapView component in index.js. It takes:
    • An array of annotations (latitude, longitude, title)
    • A region (latitude, longitude, latitudeDelta, longitudeDelta
  • Don't forget to add a style with "flex: 1"!
var React = require('react-native');

var {
  MapView,
  StyleSheet
} = React;

var MapDisplay = React.createClass({
  render() {
    var {
      latitude,
      longitude,
      title
    } = this.props;

    var annotation = [{latitude, longitude, title}];
    var region = {latitude, longitude, latitudeDelta: 1, longitudeDelta: 1};

    return (
      <MapView style={styles.map} annotations={annotation} region={region} />
    );
  }
});

var styles = StyleSheet.create({
  map: {
    flex: 1
  }
})

module.exports = MapDisplay;

Update our MapRow

  • Require "TouchableHighlight" from React Native
  • Wrap the MapRow View in a TouchableHighlight
  • Implement a simple "onPress" handler.
  • Make sure the function in the onPress handler is NOT invoked!

Transition to the Map

  • Pass the navigator prop from the list view to the row
  • Update the "onPress" handler to push a route onto the navigator.
  • Route contains title, component, and passProps.
var MapRow = React.createClass({
  selected() {
    var latitude = this.props.row.geometry.latitude;
    var longitude = this.props.row.geometry.longitude;

    this.props.navigator.push({
      title: "Map",
      component: MapDisplay,
      passProps: {
        latitude: latitude,
        longitude: longitude,
        title: "Point"
      }
    });
  },
  render() {
    var latitude = this.props.row.geometry.latitude;
    var longitude = this.props.row.geometry.longitude;
    var text = `Latitude: ${latitude}\nLongitude: ${longitude}`;

    return (
      <TouchableHighlight onPress={this.selected}>
        <View style={styles.row}>
          <Text style={styles.text}>
            {text}
          </Text>
        </View>
      </TouchableHighlight>
    );
  }
});

Step 4: Network Access

Mock Data Sux. Let's Get REAL Data!

Update Our Data Source

  • Go back to our Map Points component.
  • Let's fetch data using http.

Reminder: This is NOT Node

  • Runs on JavascriptCore on the phone
  • Node packages with native components are unlikely to work at all
  • XMLHTTPRequest shouldn't work either, because this is not a browser (except that it does through a polyfill)
  • Whatever, let's use "fetch" instead.

https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API

Let's Try That Again

  • Implement a "fetchData" method on the MapPoints component that updates the state when it is done.
  • When the component is mounted, trigger the fetchData method.
  • The "fetch" method is promise based. Note that fat arrow maintains context.
  • Call "done()" at the end of the promise chain.
var MapPoints = React.createClass({
  componentDidMount() {
    this.fetchData();
  },
  fetchData() {
    fetch(REQUEST_URL)
      .then((response => response.json()))
      .then((responseData => {
        this.setState({
          dataSource: this.state.dataSource.cloneWithRows(responseData)
        });
      }))
      .done();
  },
  ...
});

Play around with it

  • Display more data
  • Style it. Make it nicer.
  • Style the map.

React Native

By George Lee

React Native

  • 2,613