Secret shortcuts of Loading

Web Performance

Start with why

1 seconds slowness > 10% less users

The BBC found they lost an additional  10% of users for every additional second  their site took to load.

100ms faster > 1% more conversions

For Mobify, every 100ms decrease in homepage load speed  worked out to a 1.11% increase in session-based conversion,  yielding an average annual revenue increase of nearly $380,000.

20% faster > 10% more conversions

Retailer Furniture Village audited their site speed and  developed a plan to address the problems they found, leading to a 20% reduction in page load time and a 10% increase in conversion rate.

40% faster > 15% more sign-ups

Pinterest reduced perceived wait times  by 40% and this increased search  engine traffic and sign-ups by 15%.

Core Web Vitals

Largest Contenful Paint (LCP)

First Input Delay (FID)

Cumulative Layout Shift (CLS)

Performance pipeline

Loading

performance

Performance of

performance

JavaScript

Rendering

Loading Performance

Getting files you need over the network

Fetching resources over the network can be slow and expensive

Depends on

your internet speed

size of resource files

location of the server

request roundtrips

user devices

Nikola Mitrović

Development Lead & Technical Architect

Vega IT

Novi Sad, Serbia

Nikola Mitrović

Development Lead & Technical Architect

Vega IT

Novi Sad, Serbia

The cost of JavaScript

Large bundle sizes affect almost all Web Vitals metrics

JavaScript is the biggest bottleneck behind slow apps

Large bundle size will likely cause delays during

load

render

user interaction

page scroll

Optimize Loading JS

Minifying the JavaScript

Compressing the JavaScript using gzip or brotli

Removing unused code via tree shaking

Analyze bundle size with Webpack / Vite

Check lib size with Bundlephobia

Import cost VSC plugin

Use            &           attributes for third-party scripts

async

defer

Code Splitting

import React, { Suspense } from 'react';
import { BrowserRouter, Routes, Route } from 'react-router-dom';
import FirstPage from './components/FirstPage/FirstPage';
import RocketComponent from './components/Rocket/Rocket';

function App() {
  return (
    <BrowserRouter>
      <Routes>
        <Route path="/" element={<FirstPage />} />
        <Route
          path="/rocket"
          element={<RocketComponent />}
        />
      </Routes>
    </BrowserRouter>
  );
}

export default App;

Route-based Code Splitting

Route-based Code Splitting

import React, { Suspense, lazy } from 'react';
import { BrowserRouter, Routes, Route } from 'react-router-dom';
import FirstPage from './components/FirstPage/FirstPage';
const RocketLoader = lazy(() => import('./components/RocketLoader/RocketLoader'));

function App() {
  return (
    <BrowserRouter>
      <Routes>
        <Route path="/" element={<FirstPage />} />
        <Route
          path="/rocket"
          element={
            <Suspense fallback={<div>Loading...</div>}>
              <RocketLoader />
            </Suspense>
          }
        />
      </Routes>
    </BrowserRouter>
  );
}

export default App;

Component-based Code Splitting

import React, { Suspense, lazy, useRef } from 'react';
import { useVisible } from './hooks/useVisible';
import logo from './assets/images/omega.png';
import astronaut from './assets/images/astronaut.jpg';
import RocketComponent from "./components/Rocket/Rocket";
const RocketLoader = lazy(() => import("./components/RocketLoader/RocketLoader"));

function App() {
  const sectionTwoRef = useRef<HTMLDivElement>();
  const isVisible = useVisible<boolean>(sectionTwoRef);

  return (
    <>
      <div className="section section-first">
        <img src={astronaut} className="astronaut" alt="astronaut" />
      </div>
      <div ref={sectionTwoRef} className="section section-second">
         <RocketComponent />
      </div>
      {isVisible && (
        <Suspense fallback={<div>Loading...</div>}>
          <RocketLoader />
        </Suspense>
      )}
    </>
  );
}

export default App;

Reduce usage of libs

import { useState, useEffect, RefObject } from 'react';

export const useVisible = (
  ref: RefObject<HTMLDivElement>,
  options = {
    root: null,
    rootMargin: '0px',
    threshold: 1.0
  }
) => {
  const [isVisible, setIsVisible] = useState<boolean>(false);

  useEffect(() => {
    const { current: elementRef } = ref;
    
    if (elementRef) {
      const observer = new IntersectionObserver(
        ([entry]) => setIsVisible(entry.isIntersecting), options
      );

      observer.observe(elementRef);

      return () => observer.unobserve(elementRef);
    }
  }, [ref, options]);

  return isVisible;
};

Optimize Loading Images

Use              tag over background-image

Take advantage of next-gen formats

Create responsive images with

Support resolution-switching with

Load key images fast with

fetchpriority

sizes

srcset

<img>

Lazy loading & async decoding

Compress images

.my-pic {
  background-image: url("my-pic.png");
}



<img src="my-pic.png" />
<img src="large.webp" />


<img srcset="small.webp 500w, \ 
             medium.webp 1000w, \
             large.webp 2000w" 
     src="large.webp"
/>
<img src="large.webp" />


<img srcset="small.webp 400w, \ 
             large.webp 900w" 
     sizes="(max-width: 400px) 95vw, \
            (max-width: 900px) 50vw"
     src="large.webp"
/>
<img src="img.webp" />
<img src="bellow-the-fold=img.webp" />



<img src="img.webp" />
<img src="bellow-the-fold=img.webp" 
     loading="lazy" 
     decoding="async" 
/>
<img src="image.png" />
<img src="image.jpg" />


<img src="image.webp" />
<img src="image.avif" />
<img src="above-the-fold-img.webp" />
<img src="img.webp" />



<img src="above-the-fold-img.webp" 
     fetchpriority="high" 
/>
<img src="img.webp" />
import React from "react";
import RocketLoader from "./components/RocketLoader/RocketLoader";
import space from "./assets/images/space-background.jpg";
import galaxy from "./assets/images/galaxy.jpeg";

function App() {
  return (
    <div className="App">
      <img src={galaxy} className="galaxy" />
      <div className="section section-first">
        <img src={space} alt="logo" />
      </div>
      <div className="section section-second">
        <RocketLoader />
      </div>
    </div>
  );
}

export default App;

Progressive image loading (LQIP pattern)

import React from 'react';
import ProImg from './components/ProImg/ProImg';
import RocketLoader from './components/RocketLoader/RocketLoader';
import space from "./assets/images/space-background.jpg";
import galaxy from './assets/images/galaxy.jpeg';
import galaxyLight from './assets/images/galaxy-light.jpeg';

function App() {
  return (
    <div className="App">
      <ProImg
        src={galaxy}
        fallback={galaxyLight}
      />
      <div className="section section-first">
        <img src={space} alt="logo" />
      </div>
      <div className="section section-second">
        <RocketLoader />
      </div>
    </div>
  );
}

export default App;

Progressive image loading (LQIP pattern)

import React from 'react';
import ProImg from './components/ProImg/ProImg';
import RocketLoader from './components/RocketLoader/RocketLoader';
import space from "./assets/images/space-background.jpg";
import galaxy from './assets/images/galaxy.jpeg';
import galaxyLight from './assets/images/galaxy-light.jpeg';

function App() {
  return (
    <div className="App">
      <ProImg
        src={galaxy}
        fallback={galaxyLight}
      />
      <div className="section section-first">
        <img src={space} alt="logo" />
      </div>
      <div className="section section-second">
        <RocketLoader />
      </div>
    </div>
  );
}

export default App;
import React, { useState, FC } from 'react';
import ProImgProps from './ProImgProps'
import './LazyImg.css';

const ProImg = ({ src, className, alt, fallback }: ProImgProps) => {
  const [imgLoaded, setLoaded] = useState<boolean>(false);

  const handleImgLoaded = () => setLoaded(true);

  return (
    <picture className="progressive-img">
      {!imgLoaded && (
        <img
          src={fallback}
          className={`progressive-img__small ${className}`}
          alt={alt}
        />
      )}
      <img 
        src={src} 
        className={`progressive-img__large ${className}`}
        alt={alt} 
        onLoad={handleImgLoaded} 
      />
    </picture>
  );
};

export default ProImg;

Progressive image loading (LQIP pattern)

Optimize Loading Data

Reduce data payload with GraphQL

Prefetch data

<link rel="preload" href="/api/data" as="fetch" crossorigin="anonymous">

Compress data payloads

Paging and filtering

Prefetching data via useSWR

import React, { Suspense, useState } from 'react';
import { BrowserRouter, Routes, Route, Link } from 'react-router-dom';
import { mutate } from 'swr';
import FirstPage from './components/FirstPage/FirstPage';
import LaunchesPage from './components/LaunchesPage/LaunchesPage';
import RocketLoader from './components/RocketLoader/RocketLoader';


function App() {
  
  const handlePrefetch = async () => {
    const response = (await fetch('https://api.nasa.gov/planetary/apod')).json();
    await mutate('https://api.nasa.gov/planetary/apod', response);
  };

  return (
    <Suspense fallback={<RocketLoader />}>
      <BrowserRouter>
        <Routes>
          <Route
            path="/"
            element={
              <FirstPage>
                <Link to="rocket" onMouseEnter={handlePrefetch}>
                 Go to Rocket List Page
                </Link>
              </FirstPage>
            }
          />
          <Route path="/rocket" element={<LaunchesPage />} />
        </Routes>
      </BrowserRouter>
    </Suspense>
  );
}

export default App;
import React, { Suspense, useState } from 'react';
import { BrowserRouter, Routes, Route, Link } from 'react-router-dom';
import { mutate } from 'swr';
import FirstPage from './components/FirstPage/FirstPage';
import LaunchesPage from './components/LaunchesPage/LaunchesPage';
import RocketLoader from './components/RocketLoader/RocketLoader';


function App() {
  
  const handlePrefetch = async () => {
    const response = (await fetch('https://api.nasa.gov/planetary/apod')).json();
    await mutate('https://api.nasa.gov/planetary/apod', response);
  };

  return (
    <Suspense fallback={<RocketLoader />}>
      <BrowserRouter>
        <Routes>
          <Route
            path="/"
            element={
              <FirstPage>
                <Link to="rocket" onMouseEnter={handlePrefetch}>
                 Go to Rocket List Page
                </Link>
              </FirstPage>
            }
          />
          <Route path="/rocket" element={<LaunchesPage />} />
        </Routes>
      </BrowserRouter>
    </Suspense>
  );
}

export default App;
import React, { FC } from 'react';
import { useSWRConfig } from 'swr';
import { Container, Grid, Typography } from '@material-ui/core';
import FlightInfo from '../FlightInfo/FlightInfo';

const LaunchesPage: FC = () => {
  const { cache } = useSWRConfig();
  const flights = cache.get('https://api.nasa.gov/planetary/apod');

  return (
    <Container maxWidth="lg">
      <p>
        Welcome to <span>Svemirko Flights!</span>
      </p>
      <p>Explore Svemirko's recent flights</p>
      <Grid container justifyContent="flex-start" spacing={7}>
        {fligths?.map((flight) => (
          <FlightInfo key={flight.id} flight={flight} />
        ))}
      </Grid>
    </Container>
  );
};

export default LaunchesPage;

Prefetching data via useSWR

Caching

HTTP Cache eliminates network latency

Only affects "safe" HTTP methods

Cache-control headers

no-store

no-cache

max-age

Limited control

stale-while-revalidate

Content-Addressable Storage pattern

Service Workers

More control over cache

Proxy between browser and network

Different kind of scenarios

Offline capabilities + performance

Server Side Rendering

SSR Frameworks

Generate server side data at runtime

Generate server side data at runtime

Nested components with router outlet

Layouts

Stale While Revalidate cache control

Stale While Revalidate cache control

Prefetching via <Head> & <Link> components

Prefetching

Edge Computing

0kb JavaScript

Resumability

Code splitting + preloading

Lazy creation of event handlers

Serialized data

Islands architecture

Server-first

Zero JS, by default

UI agnostic

0kb JavaScript

Server side components

Serialized data

Component-specific data processing

Streams the result of data fetching

Best performance hack ever

setTimeout(() => {}, 5000);

With great power comes great responsibility

Green JavaScript

Internet is consuming 21% of the electricity

check out carbon footprint

website traffic

energy intensity

data transfers

Thank you!

n.mitrovic@vegait.rs

You can find me at

Link to the slides

Secret Shortcuts of Loading Web Performance

By nmitrovic

Secret Shortcuts of Loading Web Performance

  • 666