Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
Crossing platforms with JavaScript & React
Search
Sponsored
·
SiteGround - Reliable hosting with speed, security, and support you can count on.
→
Robert DeLuca
February 07, 2017
Technology
140
0
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
Crossing platforms with JavaScript & React
Robert DeLuca
February 07, 2017
More Decks by Robert DeLuca
See All by Robert DeLuca
Testing Big in JavaScript (Bleeding edge web)
robdel12
0
250
Testing Big in JavaScript
robdel12
0
770
Testing your app with VoiceOver
robdel12
0
83
Getting started with ember-cli-deprecation-workflow
robdel12
0
470
Thumper
robdel12
1
140
Other Decks in Technology
See All in Technology
MySQL & MySQL HeatWave Report - June 2026
freshdaz
0
130
AWS Security Hub CSPMの成功・失敗体験
cmusudakeisuke
0
550
技術・能力を向上する原理原則 #きのこセッションa #きのこ2026
bash0c7
0
130
AI-DLCを “そのまま導入しなかった”話 ~組織に合わせてアジャストした 私たちの実践共有~
hiroramos4
PRO
1
430
徹底討論!ECS vs EKS!
daitak
3
1.7k
本当の”仕事”を手放せる未来が見えた
mu7889yoon
0
130
千葉での単身赴任からAWSをやり続け、千葉に戻ってきた話
yama3133
1
120
元・セキュリティ学習経験0大学生による業務紹介 / An Introduction to the Job by a Former College Student with Zero Security Training Experience
nttcom
0
170
Deep Data Security 機能解説
oracle4engineer
PRO
2
120
週末にループ・エンジニアリングの理解を深めるためのスライド
nagatsu
0
290
AIはどのように 組織のアジリティを変えるのか?
junki
4
1.4k
クラウドファンディング版StackChan 3体(4体)をインタラクティブな体験型作品にして展示もした話 / スタックチャンお誕生日会2026
you
PRO
0
180
Featured
See All Featured
Bridging the Design Gap: How Collaborative Modelling removes blockers to flow between stakeholders and teams @FastFlow conf
baasie
0
590
Testing 201, or: Great Expectations
jmmastey
46
8.2k
Navigating Weather and Climate Data
rabernat
0
230
エンジニアに許された特別な時間の終わり
watany
107
250k
Odyssey Design
rkendrick25
PRO
2
710
The Illustrated Guide to Node.js - THAT Conference 2024
reverentgeek
1
390
Making Projects Easy
brettharned
120
6.7k
ラッコキーワード サービス紹介資料
rakko
1
3.7M
Taking LLMs out of the black box: A practical guide to human-in-the-loop distillation
inesmontani
PRO
3
2.3k
Claude Code のすすめ
schroneko
67
230k
Dominate Local Search Results - an insider guide to GBP, reviews, and Local SEO
greggifford
PRO
0
200
How Fast Is Fast Enough? [PerfNow 2025]
tammyeverts
3
620
Transcript
Crossing platforms with JavaScript & React
@robdel12
JavaScript is EVERYWHERE
Web
Server
iOS
Android
Desktop
None
What is cross platform JS?
JS that can run on more than one platform
None
“Why did the iOS team implement it like this?”
“The Android app currently doesn’t support that”
https://twitter.com/dan_abramov/status/812047645732651009
Easier to share code across many teams
More team collaboration since there’s more overlap
It allows teams to own products & not be separated
by technology
TL;DR your team now owns the iOS, Android, and (maybe)
web apps.
Consistency is
Cheaper
If you can build iOS & Android apps in the
same code base it should be cheaper
Why not bet on the web?
Native will be better than mobile web for a while
Why not take the web tooling & get native results?
You are betting on the web
Can’t beat them, join them
I decided to be ambitious
Build an Instagram clone for Web, iOS, & Android
Why an Instagram clone?
Use Impagination.js to power an Infinite scroll of images
None
Impagination will work on any JS codebase
Building infinite scroll in React Native with Impagination http://bit.ly/reactnativeinfinitescroll
We’ve already used Impagination in four different platforms
What else can be shared?
Experiment time
None
Three phases to the experiment • Planning • Implementation •
Postmortem
Planning
None
The stack • React Native • React (DOM) • Auth0
(authentication) • Graph.cool (backend) • Impagination (infinite datasets)
What should the app do? • Login / Sign up
• See your profile & images you’ve posted • Edit your profile • Post a new photo • Main list feed showing everyones posts
Web demo
Implementation
What’s the approach?
Build the web app
Start to build the native app
Realize I’ve already solved these problems in the web app
Refactor
ListPage.js
ListPage.js handles both UI & data right now
ListPage for native duplicates a lot form web ListPage
class ListPage extends React.Component { static propTypes = { data:
React.PropTypes.object, } state = { dataset: null, datasetState: null, } setupImpagination() {} componentWillMount() {this.setupImpagination();} setCurrentReadOffset = (event) => {} render () { return ( <div style={{maxWidth: "600px", margin: "0 auto", padding: "20px 0"}}> <Infinite elementHeight={ITEM_HEIGHT} handleScroll={this.setCurrentReadOffset} useWindowAsScrollContainer> {this.state.datasetState.map(record => { if (record.isPending && !record.isSettled) { return <LoadingPost key={Math.random()} />; } return <Photo key={record.content.id} photo={record.content} user={record.content.user} />; })} </Infinite> </div> ); } } const FeedQuery = gql`query($skip: Int!, $first: Int!) { allPhotos(orderBy: createdAt_DESC, first: $first, skip: $skip) { } }`; export default graphql(FeedQuery, {options: {variables: { skip: 0, first: PAGE_SIZE }}})(ListPage); Web ListPage.js
` class ListPage extends React.Component { static propTypes = {
data: React.PropTypes.object, } state = { dataset: null, datasetState: null, } setupImpagination() {} componentWillMount() {this.setupImpagination();} setCurrentReadOffset = (event) => {} render () { return ( <ScrollView style={{flex: 1}} scrollEventThrottle={300} onScroll={this.setCurrentReadOffset} removeClippedSubviews={true}> {this.state.datasetState.map(record => { if(record.isPending && !record.isSettled) { return <LoadingPost key={Math.random()}/>; } return <Photo key={record.content.id} photo={record.content} user={record.content.user} />; })} </ScrollView> ); } } const FeedQuery = gql`query($skip: Int!, $first: Int!) { allPhotos(orderBy: createdAt_DESC, first: $first, skip: $skip) { } }`; export default graphql(FeedQuery, {options: {variables: { skip: 0, first: PAGE_SIZE }}})(ListPage); Native ListPage.js
Everything but the UI is the same
New structure
Presentation & container components
<IndexRoute component={ListPageContainer} /> <Route path='feed' component={ListPageContainer} />
import ListPageView from ‘../components/presentational/ListPageView’; class ListPageContainer extends React.Component { state
= { dataset: null, datasetState: null, } setupImpagination() {} componentWillMount() {this.setupImpagination();} setCurrentReadOffset = (event) => {} render () { return ( <ListPageView setCurrentReadOffset={this.setCurrentReadOffset} datasetState={this.state.datasetState} />; ); } } const FeedQuery = gql`query($skip: Int!, $first: Int!) { allPhotos(orderBy: createdAt_DESC, first: $first, skip: $skip) { } }`; export default graphql(FeedQuery, {options: {variables: { skip: 0, first: PAGE_SIZE }}})(ListPage);
Make the container component render a separate presentation component
Leave setting the readOffset to the presentation components
setCurrentReadOffset function is passed as a prop from the container
component
t import React, { Component } from 'react'; import Infinite
from 'react-infinite'; import Photo from '../presentational/Photo'; import LoadingPost from '../presentational/LoadingPost'; const ITEM_HEIGHT = 600; const HEADER_HEIGHT = 80; class ListPageView extends Component { setCurrentReadOffset = (event) => { let currentItemIndex = Math.ceil((window.scrollY - HEADER_HEIGHT) / ITEM_HEIGHT); this.props.setCurrentReadOffset(currentItemIndex); } render() { return ( <div style={{maxWidth: "600px", margin: "0 auto", padding: "20px 0"}}> <Infinite elementHeight={ITEM_HEIGHT} handleScroll={this.setCurrentReadOffset} useWindowAsScrollContainer> {this.props.datasetState.map(record => { if (record.isPending && !record.isSettled) { return <LoadingPost key={Math.random()} />; } return <Photo key={record.content.id} photo={record.content} user={record.content.user} />; })} </Infinite> </div> ); } } export default ListPageView; Web presentation component
Native presentation component import React, { Component } from 'react';
import Photo from '../presentational/Photo'; import LoadingPost from '../presentational/LoadingPost'; import { ScrollView } from 'react-native'; const ITEM_HEIGHT = 485; class ListPageView extends Component { setCurrentReadOffset = (event) => { let currentOffset = Math.floor(event.nativeEvent.contentOffset.y); let currentItemIndex = Math.ceil(currentOffset / ITEM_HEIGHT); this.props.setCurrentReadOffset(currentItemIndex); } render() { return ( <ScrollView style={{flex: 1}} scrollEventThrottle={300} onScroll={this.setCurrentReadOffset} removeClippedSubviews={true}> {this.props.datasetState.map(record => { if(record.isPending && !record.isSettled) { return <LoadingPost key={Math.random()}/>; } return <Photo key={record.content.id} photo={record.content} user={record.content.user} />; })} </ScrollView> ); } } export default ListPageView;
This theme continues throughout the entire app
<IndexRoute component={ListPageContainer} /> <Route path='feed' component={ListPageContainer} /> <Route path='new' component={CreatePostContainer}
onEnter={this.requireAuth.bind(this)} /> <Route path='signup' component={CreateUserContainer} /> <Route path='profile' component={UserProfileContainer} > <IndexRoute component={UserProfileContainer} /> <Route path='edit' component={EditProfileContainer} /> </Route> <Route path='logout' component={() => <Logout logout={this.handleToken.bind(this)} /> } />
Native app demo
Postmortem
Building the apps in time was hard…
None
None
Figuring out what code is shareable
Figuring out how to make that code shareable
React Router is neat & works cross platform There are
different imports for React Native & React
None
Auth0 was very easy to implement on both platforms. There
are different APIs for React Native & React
AsyncStorage vs localStorage
What all ended up being shared?
✅ List feed ✅ User profile ✅ Edit user profile
✅ Sign up ✅ New post
Beyond login mostly everything else is the same
The UI changed but not the business logic
Key takeaways
We’re in a post DOM world
Write JavaScript interaction models
The UI framework will change but the underlying model driving
it won’t
“It’s just JavaScript”
We get stronger libraries by increasing the number of users
& contributors.
React makes this very easy thanks to React & React
Native
I was able to share the same five container components
across three different platforms
Write one container component and many UI components
The core of this app is shared
That’s a cost savings
I own this entire product & its 3 platforms
In 2 weeks I was able do all of this
Cross platform JS FTW
Instagram also agrees with me https://engineering.instagram.com/react-native-at-instagram- dd828a9a90c7#.i364vchox
None
If this kind of stuff interests you
We’re hiring!
Thanks! @robdel12