The Fastest Code
is the code you don't load
How not to get yelled at by Alex Russell
Smaller
Smarter
Overview
Tooling
- Tree shaking
- SSR
- Code splitting
Environment
- Less code
- Browser Devtools
- Browser APIs
- Network features
Smaller
data:image/s3,"s3://crabby-images/c446a/c446a87bde88d19b822ebfd17a3417fee219525b" alt=""
Smaller Frameworks
Smaller Frameworks
data:image/s3,"s3://crabby-images/e50ed/e50ed7e72c96c23e79a5d8bd3747002291deb29b" alt=""
Smarter
Environment Edition
Browser Devtools
- Inspector
- Lighthouse
Inspector: CSS+JS Coverage
data:image/s3,"s3://crabby-images/dcd05/dcd0507495bc247c7319886360ef73e07b59e45a" alt=""
Browser Devtools
data:image/s3,"s3://crabby-images/5e85b/5e85b78912775ca96ba14530b4c26de5d73877a7" alt=""
Inspector: CSS Coverage
Browser Devtools
Lighthouse
Browser Devtools
data:image/s3,"s3://crabby-images/73730/73730a9383622a8d850dd833152ebb9c8cdf482b" alt=""
Browser APIs
- Service Worker
- IntersectionObserver
Intersection Observer
data:image/s3,"s3://crabby-images/d2481/d248129a01cc6c0a21d089e555e1f5f2245cb19d" alt=""
Browser APIs
Intersection Observer: Native DOM API
Browser APIs
var iObserver = new IntersectionObserver((entries) => {
// If intersectionRatio is 0, the target is out of view
if (entries[0].intersectionRatio <= 0) return;
loadItems(10);
});
// start observing
iObserver.observe(
document.querySelector('.scrollerFooter')
);
Intersection Observer: react-intersection-observer
Browser APIs
import Observer from 'react-intersection-observer'
<Observer>
{(inView) => <h2>{`Header inside viewport ${inView}.`}</h2>}
</Observer>
data:image/s3,"s3://crabby-images/c0dff/c0dfff2f685409d4dee46b792b226f66412c3e8b" alt=""
Service Worker: offline-first pattern
Browser APIs
Service Worker: offline-first pattern
Browser APIs
Step 1: Configure plugin in webpack.config.js:
// webpack.config.js
var OfflinePlugin = require('offline-plugin');
module.exports = {
// ...
plugins: [
// ... other plugins
// Should usually be last
new OfflinePlugin({
...
})
]
}
Service Worker: offline-first pattern
Browser APIs
// main.entry.js
require('offline-plugin/runtime').install();
runtime.install({
onUpdateReady: () => {
// Tells to new SW to take control immediately
runtime.applyUpdate();
},
onUpdated: () => {
// Reload page to load the new version
window.location.reload();
},
onUpdateFailed: () => {
console.log('SW Event:', 'onUpdateFailed');
}
});
Step 2: Add the runtime into your entry file
Network features
- HTTP2
- Preload
data:image/s3,"s3://crabby-images/32315/32315c08cf853d7dee86a501b6d67116793eb63a" alt=""
Network features
H2
Network features
Server Push
https://jakearchibald.com/2017/h2-push-tougher-than-i-thought/
data:image/s3,"s3://crabby-images/b3ce5/b3ce5c8f38a89ddcb512b6977e9193496d06230c" alt=""
Smarter
Tooling Edition
Code Splitting
- Per page
- Dynamic
Code splitting
data:image/s3,"s3://crabby-images/aa8eb/aa8eb55b40b809ac9b0dd512fcc6e4c0fabdcb3d" alt=""
Per entry-point
data:image/s3,"s3://crabby-images/1fff3/1fff30f1cdde16d6f554400b053583df5656152b" alt=""
data:image/s3,"s3://crabby-images/c115c/c115c09e10264efc895aa4dda4f9d6c5a8226b3e" alt=""
Dynamic - component-based
Code splitting
data:image/s3,"s3://crabby-images/094da/094da26488f9402e9736bad47b8e84da59ad9abd" alt=""
Code splitting
webpack-bundle-analyzer: stats.json
669kB
83kB
80kB
489kB
Code splitting
webpack-bundle-analyzer: gzipped
data:image/s3,"s3://crabby-images/5957a/5957ae3b7c475e59fbcff21f4f3446115b6424a5" alt=""
61kB
7kB
47kB
11kB
Code splitting
webpack-bundle-analyzer: parsed
data:image/s3,"s3://crabby-images/cc8a2/cc8a2cbc26092bd67d7ec4ed737e677ff601886a" alt=""
151kB
48kB
209kB
25.4 kB
Code splitting: set-up
.babelrc
{
"presets": [
["env", {"modules": false}],
"react",
"stage-2"
],
"plugins": [
"transform-flow-strip-types"
],
"env": {
"browser": {
"presets": [
["env", {
"targets": {"chrome": 57, "firefox": 52, "safari": 10},
"modules": false,
"useBuiltIns": true
}],
"react",
"stage-2"
]
}
}
}
Code splitting: device-based
react-loadable
import Loadable from 'react-loadable';
const RadarLoader = Loadable({
loader: () => import(/* webpackChunkName: "radar" */ './radar'),
LoadingComponent: RadarSpinner
});
const RadarSpinner = ({ isLoading, error, pastDelay }: Props) => {
if (isLoading) {
return pastDelay ? <Spinner>Loading...</Spinner> : null;
} else if (error) {
return <div>Error! Component failed to load</div>;
}
return null;
};
export default class extends Component {
render() {
return (
<Container>
<Media query={d.mqs.tablet}>
{matches => (matches ? <RadarLoader /> : null)}
</Media>
</Container>
);
}
}
1
2
Code splitting: interaction-based
react-loadable
let Db = Loadable({
loader: () => import(/* webpackChunkName: "searchdb" */ './db'),
LoadingComponent: DbShim
});
class Search extends Component {
state = {initialised: false, query: ''};
loadDB = () => {
if (this.state.initialised) return;
this.setState({ initialised: true });
Database.preload();
};
updateQuery = (e) => this.setState({ query: e.target.value; });
render() {
const {query} = this.state;
const {cards, submitSearch} = this.props;
return (
<Container>
<Form query={query} onFocus={this.loadDB} onChange={this.updateQuery} />
<Db cards={cards} term={query} onTermUpdated={submitSearch} />
</Container>
);
}
}
3
1
2
Preload chunks
preload-webpack-plugin
data:image/s3,"s3://crabby-images/8616a/8616a6505e3e929b65d0e661b691d369074f2e7e" alt=""
Preload chunks
preload-webpack-plugin
<!DOCTYPE html>
<html lang="en">
<head>
...
<link rel="preload" as="script" href="/js/chunks/radar.js">
<link rel="preload" as="script" href="/js/chunks/root.js">
<link rel="preload" as="script" href="/js/chunks/searchdb.js">
<link rel="preload" as="script" href="/js/chunks/smoothscroll.js">
<style type="text/css">...</style>
<link href="/css/app.css" rel="preload" as="style" onload="this.rel='stylesheet'">
</head>
<body>...</body>
</html>
data:image/s3,"s3://crabby-images/c115c/c115c09e10264efc895aa4dda4f9d6c5a8226b3e" alt=""
👇
Thanks!
@oliverturner
Presentation to MancReact May 2017
By Oliver Turner
Presentation to MancReact May 2017
A breakdown of the various tools and techniques we can use to slim down our payloads and keep our users (and Alex Russell) happy.
- 897