Superpowers of browser's Web API
Web API
console.log(1);
setTimeout(() => console.log(2), 0);
console.log(3);
data:image/s3,"s3://crabby-images/ccebd/ccebdaec23edd1fcc323b098631a6ccb518b3b0c" alt=""
?
Web API
Collection of built-in interfaces that allow developers to interact with and manipulate web pages and web applications within a web browser.
data:image/s3,"s3://crabby-images/7aa1e/7aa1ee8b5aa1eaf7e8fc722d2c97c11ab6ff6b72" alt=""
- Web API docs
Web APIs
Nikola Mitrović
Development Lead & Software Engineer
Vega IT
Novi Sad, Serbia
data:image/s3,"s3://crabby-images/63f92/63f9254a174e6e87f2dff0a36981e610e321a1da" alt=""
data:image/s3,"s3://crabby-images/4f44b/4f44bbd183cdd047406cfe128e93d42f8434e864" alt=""
data:image/s3,"s3://crabby-images/704be/704be572709421081b3a7185a71e4a0486dcb2d6" alt=""
Intersection Observer API
-
Provides a way to asynchronously observe changes in the intersection of a target element with an ancestor element or with a top-level document's viewport
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;
};
data:image/s3,"s3://crabby-images/d8310/d8310e0de7e34780367a94894b51ad6a05cd39a9" alt=""
data:image/s3,"s3://crabby-images/3a8e8/3a8e880e4cd671d61f9ea63ee634e6b6e75d3884" alt=""
data:image/s3,"s3://crabby-images/49323/49323e3aea671d69f407277d9369974e456be1f4" alt=""
data:image/s3,"s3://crabby-images/07ecc/07ecc43d37df24e02da2337f2c61293f2ab92bde" alt=""
data:image/s3,"s3://crabby-images/2e1a8/2e1a877363ff1b437067867809022ebf65384ac4" alt=""
data:image/s3,"s3://crabby-images/0a39d/0a39d62a96a4d7ab672df11f10ed70bb69a8e1cb" alt=""
Intersection Observer API
Lazy-load images
Infinite scroll lists
Defer animations
Save compute power
03
02
04
01
Intersection Observer API - Use Cases
Intersection Observer API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Page Visibility API
-
Provides events you can watch for to know the current visibility state of the page
-
Triggers when user
switches tab
minimizes window
puts another window as overlay
locks/turns off device
Page Visibility API
data:image/s3,"s3://crabby-images/54578/54578e95cb212c327239e3d71be95f8293c4a466" alt=""
import { useEffect, useState } from "react";
export const usePageVisibility = () => {
const [pageVisible, setPageVisible] = useState<boolean>(true);
useEffect(() => {
const handleVisibilityChange = () => {
const isPageVisible = document.visibilityState === "visible";
setPageVisible(isPageVisible);
};
document.addEventListener("visibilitychange", handleVisibilityChange);
return () => {
document.removeEventListener("visibilitychange", handleVisibilityChange);
};
}, []);
return pageVisible;
};
data:image/s3,"s3://crabby-images/af4c0/af4c0881170eb6ea9aaaa0975e6a4af879af86fd" alt=""
data:image/s3,"s3://crabby-images/6f9f1/6f9f12ebf4c3451651ec7500867c1ca7ccd205fb" alt=""
Page Visibility API - Use Cases
Improving performance
01
Pause audio/video
02
Detect page visit count
03
Poll server data
04
Carousel
05
Page Visibility API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Screen Wake Lock API
-
Prevent devices from locking the screen when an application needs to keep running
-
Only visible (active) documents can acquire the screen wake lock
Screen Wake Lock API
import { useEffect, useCallback, useRef } from 'react';
export const useWakeLock = (shouldWakeLock = false) => {
let sentinel = useRef<WakeLockSentinel | null>(null);
const requestWakeLockSentinel = useCallback(async () => {
sentinel.current = await navigator.wakeLock.request('screen');
}, []);
const releaseWakeLockSentinel = useCallback(async () => {
await sentinel.current?.release();
sentinel.current = null;
}, []);
useEffect(() => {
(async () => {
if (shouldWakeLock) {
await requestWakeLockSentinel();
return releaseWakeLockSentinel;
}
await releaseWakeLockSentinel();
})();
}, [shouldWakeLock, requestWakeLockSentinel, releaseWakeLockSentinel]);
return { isWakeLock: !!sentinel.current };
};
data:image/s3,"s3://crabby-images/21999/219996203ed3490146f834585ec58d6d43f08f04" alt=""
data:image/s3,"s3://crabby-images/4cb98/4cb980e7b9a7b281d29c6e581b6a897b0d4914c7" alt=""
data:image/s3,"s3://crabby-images/abe03/abe03c238586ed36fabb6db6939141c85412ca5a" alt=""
Screen Wake Lock API - Use Cases
Reading an ebook
Presenting to an audience
Following a recipe
Map navigation
Scanning a QR / barcode
03
05
02
04
01
Screen Wake Lock API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Screen Wake Lock API - Drawbacks
Network API
-
Provides information about the system's connection, such as the current bandwidth of the user's device or whether the connection is metered
-
Detect connection changes
import { useState, useEffect } from "react";
enum ENetworkEffectiveType {
SLOW_TWO_G = "slow-2g",
TWO_G = "2g",
THREE_G = "3g",
FOUR_G = "4g",
}
export const useNetwork = () => {
const [isFastNetwork, setIsFastNetwork] = useState<boolean | null>(null);
useEffect(() => {
if ("connection" in navigator) {
const { connection: { effectiveType } } = navigator;
setIsFastNetwork(effectiveType === ENetworkEffectiveType.FOUR_G);
}
}, []);
return isFastNetwork;
};
data:image/s3,"s3://crabby-images/ca2e5/ca2e593713d8852d8e1437ee7df29db2479efc53" alt=""
data:image/s3,"s3://crabby-images/cd498/cd4987eb33d4a5943824124047e6aa6e0bebfc85" alt=""
Network API
import { useState, useEffect } from "react";
enum ENetworkEffectiveType {
SLOW_TWO_G = "slow-2g",
TWO_G = "2g",
THREE_G = "3g",
FOUR_G = "4g",
}
export const useNetwork = () => {
const [isFastNetwork, setIsFastNetwork] = useState<boolean | null>(null);
const updateConnectionStatus = ({ target: { effectiveType } }: Event) => {
setIsFastNetwork(effectiveType === ENetworkEffectiveType.FOUR_G);
};
useEffect(() => {
if ("connection" in navigator) {
const { effectiveType } = navigator.connection;
setIsFastNetwork(effectiveType === ENetworkEffectiveType.FOUR_G);
navigator.connection.addEventListener("change", updateConnectionStatus);
}
}, []);
return isFastNetwork;
};
data:image/s3,"s3://crabby-images/ca2e5/ca2e593713d8852d8e1437ee7df29db2479efc53" alt=""
Network API
Network API - Use Cases
Notification to the user
Preload large resources
02
01
Network API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Background Sync API
-
Provides a way to defer tasks to be run in a service worker until the user has a stable network connection
Background Sync API
import React from "react";
import { Link } from "react-router-dom";
export const FlightsLink = () => {
const registerPlanetsSync = async () => {
const registration = await navigator.serviceWorker.ready;
try {
await registration.sync.register("sync-planets");
} catch {
console.log("Background Sync could not be registered!");
}
};
return (
<Link
to="rocket"
onMouseEnter={registerPlanetsSync}
>
EXPLORE
</Link>
)
}
data:image/s3,"s3://crabby-images/abf2d/abf2d331ed8fc1d2dfe0f72418fa5190f7fc26aa" alt=""
data:image/s3,"s3://crabby-images/038c0/038c0dce836d3c01d2242afd63acc3685bce6d6e" alt=""
data:image/s3,"s3://crabby-images/c2715/c27158f23c0f2edc7de1fd41a18c7476741d51ff" alt=""
data:image/s3,"s3://crabby-images/a31b0/a31b02524c5daae9b079a35c5e8c2066e2574eb2" alt=""
data:image/s3,"s3://crabby-images/e750d/e750d888d5f913a35ff407baa5cf696854610fa5" alt=""
Background Sync API
import React from "react";
import { Link } from "react-router-dom";
export const FlightsLink = () => {
const registerPlanetsSync = async () => {
const registration = await navigator.serviceWorker.ready;
try {
await registration.sync.register("sync-planets");
} catch {
console.log("Background Sync could not be registered!");
}
};
return (
<Link
to="rocket"
onMouseEnter={registerPlanetsSync}
>
EXPLORE
</Link>
)
}
self.addEventListener("sync", (event) => {
if (event.tag === "sync-planets") {
event.waitUntil(fetchPlanets());
}
});
data:image/s3,"s3://crabby-images/c2715/c27158f23c0f2edc7de1fd41a18c7476741d51ff" alt=""
data:image/s3,"s3://crabby-images/29870/298701d7f700e75764bdda04a39a3e8e5c2f989d" alt=""
Background Sync API - Use Cases
Offline UX
Send email while offline
02
01
Background Sync API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Background Sync API
import React from "react";
import { Link } from "react-router-dom";
export const FlightsLink = () => {
const registerPlanetsSync = async () => {
const registration = await navigator.serviceWorker.ready;
try {
await registration.sync.register("sync-planets");
} catch {
console.log("Background Sync could not be registered!");
}
};
return (
<Link
to="rocket"
onMouseEnter={registerPlanetsSync}
>
EXPLORE
</Link>
)
}
const fetchPlanets = async () => {
const planetsResponse = await fetch(
"https://exoplanetarchive.ipac.caltech.edu"
);
const planets = await planetsResponse.json();
};
self.addEventListener("sync", (event) => {
if (event.tag === "sync-planets") {
event.waitUntil(fetchPlanets());
}
});
data:image/s3,"s3://crabby-images/29870/298701d7f700e75764bdda04a39a3e8e5c2f989d" alt=""
data:image/s3,"s3://crabby-images/c2715/c27158f23c0f2edc7de1fd41a18c7476741d51ff" alt=""
data:image/s3,"s3://crabby-images/4486b/4486be329a4bacfa9a3bfd7805cc3da0a0d9a3aa" alt=""
data:image/s3,"s3://crabby-images/1ff02/1ff0263b6edd152063db99c7a3abd0dae12183d7" alt=""
Broadcast Channel API
-
Allows basic communication between browsing contexts (that is, windows, tabs, frames, or iframes) and workers on the same origin
-
Bi-directional communication
import React from "react";
import { Link } from "react-router-dom";
export const FlightsLink = () => {
const registerPlanetsSync = async () => {
const registration = await navigator.serviceWorker.ready;
try {
await registration.sync.register("sync-planets");
} catch {
console.log("Background Sync could not be registered!");
}
};
return (
<Link
to="rocket"
onMouseEnter={registerPlanetsSync}
>
EXPLORE
</Link>
)
}
const broadcast = new BroadcastChannel("planets-channel");
const fetchPlanets = async () => {
const planetsResponse = await fetch(
"https://exoplanetarchive.ipac.caltech.edu"
);
const planets = await planetsResponse.json();
broadcast.postMessage({ planets });
};
self.addEventListener("sync", (event) => {
if (event.tag === "sync-planets") {
event.waitUntil(fetchPlanets());
}
});
data:image/s3,"s3://crabby-images/c2715/c27158f23c0f2edc7de1fd41a18c7476741d51ff" alt=""
data:image/s3,"s3://crabby-images/29870/298701d7f700e75764bdda04a39a3e8e5c2f989d" alt=""
data:image/s3,"s3://crabby-images/87779/87779cdd73a1d90408b77f80be290763756ab625" alt=""
data:image/s3,"s3://crabby-images/d9b79/d9b7904bbc3f9867321cc11af719aa837325de73" alt=""
Broadcast Channel API
import React from "react";
import { Link } from "react-router-dom";
export const FlightsLink = () => {
useEffect(() => {
const broadcast = new BroadcastChannel("planets-channel");
broadcast.onmessage = ({
data: { planets }
}) => setPlanets(planets);
return () => broadcast.close();
}, []);
const registerPlanetsSync = async () => {
const registration = await navigator.serviceWorker.ready;
try {
await registration.sync.register("sync-planets");
} catch {
console.log("Background Sync could not be registered!");
}
};
return (
<Link
to="rocket"
onMouseEnter={registerPlanetsSync}
>
EXPLORE
</Link>
)
}
const broadcast = new BroadcastChannel("planets-channel");
const fetchPlanets = async () => {
const planets = await fetch(
"https://exoplanetarchive.ipac.caltech.edu"
).then((response) => response.json());
broadcast.postMessage({ planets });
};
self.addEventListener("sync", (event) => {
if (event.tag === "sync-planets") {
event.waitUntil(fetchPlanets());
}
});
data:image/s3,"s3://crabby-images/c2715/c27158f23c0f2edc7de1fd41a18c7476741d51ff" alt=""
data:image/s3,"s3://crabby-images/29870/298701d7f700e75764bdda04a39a3e8e5c2f989d" alt=""
data:image/s3,"s3://crabby-images/ab800/ab800af7f050cadde0095b7451c58fe94a7c8a11" alt=""
Broadcast Channel API
Detect user action
Logout in all tabs
02
01
Broadcast Channel API - Use cases
Broadcast Channel API - Browser Support
data:image/s3,"s3://crabby-images/ebccd/ebccd3db6964f64f9f3931439cb58b3df69c82c0" alt=""
data:image/s3,"s3://crabby-images/75a19/75a19a46301f9655dfba98341b66e57f64f26566" alt=""
data:image/s3,"s3://crabby-images/1f48c/1f48c77eb5ef66184ddcd4356185181e1299940e" alt=""
Web API - Bonus points
-
Beacon API
-
Web Speech API
-
Page Visibility API
-
Network API
-
Web Share API
-
Background Fetch API
-
Web Authn
Web API - Takeaways
Non-existing browser support
Experimental browser support
Device support
03
02
01
04
Typescript support
Web API - Takeaways
Standardized code
Standardized code
Standardized code
Smooth learning curve
Increase performance
03
05
02
04
01
With great power comes great responsibility
Green JavaScript
data:image/s3,"s3://crabby-images/c6c64/c6c64b37bf6504fa6fd146ec2674cdd10c35de5c" alt=""
Internet is consuming 21% of the electricity
check out carbon footprint
website traffic
energy intensity
data transfers
data:image/s3,"s3://crabby-images/f10e5/f10e55d81406d82ed38594080ee6ac3c8634b889" alt=""
Thank you!
n.mitrovic@vegait.rs
You can find me at
Link to the slides
https://github.com/nmitrovic-vega/superpowers-web-api
data:image/s3,"s3://crabby-images/d4084/d40849ca9fd9e0a8140e43a435fe195d4355838d" alt=""
Superpowers of browser's Web APIs
By nmitrovic
Superpowers of browser's Web APIs
- 340