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
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
【2026年版】 ベクトル検索とEmbedding最前線
mocobeta
24
7.5k
GitHub Copilot 最新アップデート – 「一歩先」の実践活用術
moulongzhang
5
1.7k
いまさら聞けない「仕様駆動開発入門」 〜AI活用時代の開発プロセスを考える〜
findy_eventslides
2
200
Flow 不死:AI 時代 DevOps 的不變本質
cheng_wei_chen
2
510
入門!AWS Blocks
ysuzuki
1
190
事業会社における 機械学習・推薦システム技術の活用事例と必要な能力 / ml-recsys-in-layerx-wantedly-2026
yuya4
0
160
クレデンシャル流出 ― 攻撃 3 時間 vs 復旧 10 時間。この非対称性にどう備えるか
kazzpapa3
3
560
5分でわかるDuckDB Quack
chanyou0311
3
250
気軽に使える"情報のハブ"としてのNotion活用 〜フロー情報の集積点 と、 Claude Code × Notion AI〜
syucream
1
200
[チョークトーク資料]AWS DevOps Agent を使いこなす / AWS Dev Ops Agent Chalk Talk AWS Summit Japan 2026
kinunori
4
770
時期が悪い!それでもRaspberry Piを買って遊んで活用するには / 20260627-osc26do-rpi-jikigawarui
akkiesoft
0
820
コミットの「なぜ」を読む
ota1022
0
120
Featured
See All Featured
Understanding Cognitive Biases in Performance Measurement
bluesmoon
32
2.9k
Redefining SEO in the New Era of Traffic Generation
szymonslowik
1
340
Become a Pro
speakerdeck
PRO
31
6k
Jamie Indigo - Trashchat’s Guide to Black Boxes: Technical SEO Tactics for LLMs
techseoconnect
PRO
0
180
KATA
mclloyd
PRO
35
15k
Why You Should Never Use an ORM
jnunemaker
PRO
61
9.9k
Navigating Algorithm Shifts & AI Overviews - #SMXNext
aleyda
1
1.3k
Getting science done with accelerated Python computing platforms
jacobtomlinson
2
240
Making the Leap to Tech Lead
cromwellryan
135
9.9k
Optimising Largest Contentful Paint
csswizardry
37
3.7k
Leveraging Curiosity to Care for An Aging Population
cassininazir
1
270
世界の人気アプリ100個を分析して見えたペイウォール設計の心得
akihiro_kokubo
PRO
72
40k
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