Progressive Web Apps (PWA)

A Progressive Web App (PWA) uses HTML/CSS/JavaScript to build a web application that can run on mobile devices (as well as computers). PWAs have the same look-and-feel as native mobile device apps.

PWAs are HTML/CSS/JavaScript websites that have some additional features. A PWA can:

 

A PWA can only run on secure HTTPS websites or from localhost. The localhost is only included to allow developers to test their PWA locally before uploading it to a HTTPS server.

 

A PWA should be responsive. They should display properly on mobile devices, laptops and PCs.

A PWA is more discoverable than a native app because it is available on the www rather than an app store. A PWA can be easily shared by sending a link.

Manifest

In order to turn a webpage into a PWAr, it must link to a manifest file. A manifest file is a JSON file that defines how a PWA will behave when it is installed on a mobile device or computer.

A manifest file can be of type .json or .webmanifest. We shall use .json in these notes.

The complete set of properties that can be included in the manifest file can be found at this link. The main properties in the manifest file are:
name (required)
This is PWA's name. It appears under the PWA's icon when the is installed on a mobile device or computer.
short_name (optional)
short_name provides a shortened name that can be used when there is insufficient space on a mobile device to display the name attribute.
start_url (required)
This is the first page that will be shown to the user when the PWA opens. Normally, this will be index.html.
icons (required)
This is an array of filenames that hold icon images. A PWA can be installed on phone, tablets, laptops or computer. The icons array allows us use different images and image sizes for the icon, depending on the screen size. There must be at least one icon in the array. If there is more than one icon in the array, then the mobile device or computer will decide which icon it will use. Icons should be set to be maskable. Making an icon maskable means that the image will be automatically scalled to fit the icon shape for the device that it is being diaplyed on. For example, most mobile phones show icons with a rounded border and containing has some padding.
display (required)
This specifies how the PWA should be displayed on the mobile device or computer.
standalone
The application will look and feel like a standalone application.
fullscreen
All of the available screen will be used.
minimal-ui
The application will look and feel like a standalone application, but it will also contain a minimal set of UI elements that can be used to control navigation. The UI elements will vary by browser.
browser
The application will open in a browser tab or a new window.
description (optional)
This describes what the PWA does.
background_color (optional)
This is the background colour of the PWA's icon background and splash screen background.
theme_color (optional)
It will be the background color of the PWA's status bar.
orientation (optional)
This is orientation to have when displaying the app. Three commonly used orientation values are:

An example of a manifest file is shown below:

manifest.json

{
  "name": "Offline First Example",
  "short_name": "Offline",
  "description": "This implements an offline first policy when loading a PWA's assets.",
  "start_url": "./",
  "scope": ".",
  "display": "standalone",
  "background_color": "white",
  "theme_color": "white",
  "orientation": "any",
  "icons": [{
    "src": "icons/icon_small_red.png",
      "sizes": "16x16 64x64",
      "type": "image/png"
    }, {
      "src": "icons/icon_medium_green.png",
      "sizes": "128x128 512x512",
      "type": "image/png"
    }, {
      "src": "icons/icon_large_blue.png",
      "sizes": "1024x1024",
      "type": "image/png",
      "purpose": "maskable"
    }]
}

You can use a manifest generator, such as https://www.simicart.com/manifest-generator.html/ to generate the icons and other basic manifest file data.

The code below needs to be added at the top of the index.html file. As well as linking to the manifest.json file, the code below also includes some other links that are needed when installing a PWA on an iPhone or iPad.

<link rel="manifest" href="manifest.json">

<!-- ios support -->        
<link rel="icon" href="icons/icon_small_red.png" type="image/png">
<link rel="apple-touch-icon" href="icons/icon_medium_green.png">   
<meta name="msapplication-TileImage" content="icons/icon_large_blue.png">
<meta name="msapplication-TileColor" content="#FFFFFF">
<meta name="theme-color" content="white">  
<meta name="mobile-web-app-capable" content="yes">
<meta name="apple-mobile-web-app-status-bar-style" content="black">
<meta name="apple-mobile-web-app-title" content="Offline First Example">

Service Workers

In order to use a service worker, the landing webpage (index.html) needs to register a service worker, as shown below. In the code below, the service worker called sw_offline_first.js will be loaded.

        <!-- Register the service worker in the landing webpage. The registration only needs to happen once. Do not copy this code into other HTML files -->
        <script>
            window.addEventListener('load', () =>
            {
                if ("serviceWorker" in navigator)
                {
                    navigator.serviceWorker.register("sw_offline_first.js")
                }
            })
        </script>

 

A complete index.html file that includes the manifest, ios support, and registers a service worker is shown below:

index.html

<!doctype html>
<html>
    <head>
        <title>Course notes example code</title>
        <meta charset="utf-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <link rel="icon" type="image/png" href="icons/icon_small_red.png">

        <link rel="stylesheet" href="css/style.css">

        <link rel="manifest" href="manifest.json">

        <!-- ios support -->
        <link rel="icon" href="icons/icon_small_red.png" type="image/png">  
        <link rel="apple-touch-icon" href="icons/icon_medium_green.png">   
        <meta name="msapplication-TileImage" content="icons/icon_large_blue.png">
        <meta name="msapplication-TileColor" content="#FFFFFF">
        <meta name="theme-color" content="white">  
        <meta name="mobile-web-app-capable" content="yes">  
        <meta name="apple-mobile-web-app-status-bar-style" content="black"> 
        <meta name="apple-mobile-web-app-title" content="Offline First Example">

        <!-- Register the service worker in the landing webpage. The registration only needs to happen once. Do not copy this code into other HTML files -->
        <script>
            window.addEventListener('load', () =>
            {
                if ("serviceWorker" in navigator)
                {
                    navigator.serviceWorker.register("sw_offline_first.js")
                }
            })
        </script>
    </head>

    <body class="fullscreen">
        <div id="container">
            <h1 class="title">Offline-first Homepage</h1>
            <a href="non_cached_webpage.html">Open non-cached webpage</a>
        </div>
    </body>
</html>

A service worker:

Service workers must be placed in the root folder, so that their scope includes your entire website.

 

An example service worker is given below.

sw_offline_first.js

const cacheName = 'offline_pwa_example_version_1.0'

const filesToCache =
        ['manifest.json',
            'index.html',
            'offline_message.html',
            'css/style.css',
            'icons/icon_small_red.png',
            'icons/icon_medium_green.png',
            'icons/icon_large_blue.png']


// Install the service worker and cache the files in the array filesToCache[] 
self.addEventListener('install', e =>
{
    e.waitUntil(caches.open(cacheName)
            .then(cache =>
            {
                cache.addAll(filesToCache)
                return true
            }))
})


// Delete old versions of the cache when a new version is first loaded 
self.addEventListener('activate', event =>
{
    event.waitUntil(caches.keys()
            .then(keys => Promise.all(keys.map(key =>
                {
                    if (!cacheName.includes(key))
                    {
                        return caches.delete(key)
                    }
                }))))
})


// Fetch offline cached first, then online, then offline error page 
self.addEventListener('fetch', function (e)
{
    e.respondWith(caches.match(e.request)
            .then(function (response)
            {
                if (response) // file found in cache
                {
                    return response
                } 
                else // file found online
                {
                    return fetch(e.request)
                }
            })
            .catch(function (e)
            {
                // offline and not in cache
                return caches.match('offline_message.html')
            })
            )
})
const cacheName = 'offline_pwa_example_version_1.0'

The cacheName allows for updates to be made at a later date to the PWA.

const filesToCache =
        ['manifest.json',
            'index.html',
            'offline_message.html',
            'css/style.css',
            'icons/icon_small_red.png',
            'icons/icon_medium_green.png',
            'icons/icon_large_blue.png']

The filesToCache holds the list of files that we want to store in the browser's cache.

// Install the service worker and cache the files in the array filesToCache[] 
self.addEventListener('install', e =>
{
    e.waitUntil(caches.open(cacheName)
            .then(cache =>
            {
                cache.addAll(filesToCache)
                return true
            }))
})

When the service work first installs, it saves the filesToCache in the browser's cache.

Whenever the user opens the PWA in future, the saved files in the browser's cache will be used. This means the files only have to be downloaded once.

// Delete old versions of the cache when a new version is first loaded 
self.addEventListener('activate', event =>
{
    event.waitUntil(caches.keys()
            .then(keys => Promise.all(keys.map(key =>
                {
                    if (!cacheName.includes(key))
                    {
                        return caches.delete(key)
                    }
                }))))
})

If the cacheName is changed, then the old set of cached files will be deleted and the new set of files in filesToCache will be downloaded. This allows us to update the content of the PWA.

// Fetch offline cached first, then online, then offline error page 
self.addEventListener('fetch', function (e)
{
    e.respondWith(caches.match(e.request)
            .then(function (response)
            {
                if (response) // file found in cache
                {
                    return response
                } 
                else // file found online
                {
                    return fetch(e.request)
                }
            })
            .catch(function (e)
            {
                // offline and not in cache
                return caches.match('offline_message.html')
            })
            )
})

With an offline-first approach, the browser will firstly try to load files from the browser cache (using caches.match(e.request)). If this fails, it will download the file from the server (using fetch(e.request)). If this fails, then it will load the file offline_message.html

The above example implements an offline-first policy. When a user goes to run the PWA, the service worker will try to retrieve files from the browser's cache. It will only download files if they are not in the browser's cache.

A complete offline-first PWA example that uses the code described above can be found at this this link.

A .zip file containing this PWA can be downloaded from here.

Search online and list examples of when it is best to use an offline-first approach and when it is best to use an online-first approach.

Adjust the offline_first code above to make an online-first PWA, as shown here. The solution as a .zip file can be downloded from here.

Lighthouse

The Chrome Lighthouse tab can be used to test if a website is a valid PWA. We can use the Lighthouse to check how compliant our app is to the PWA requuirements. Lighthouse is accessed via the inspect (F12) option. It is on the same menu as the the Console tab.

Use Lighthouse to analyse the offline first PWA example above.

Application

The Chrome Application tab can be used to test if a website is a PWA. We can use the Application tab to check is a PWA is registered, to unregister a PWA, view all registered PWAs and simulate the browser being offlie. The Application tab is accessed via the inspect (F12) option. It is on the same menu as the the Console tab.

Open your browser's code inspector (F12). Use the Manifest option in the Application tab to see that offline_first has been registered as a PWA.

Use the Application tab to simulate the offline_first PWA being offline.

 
<div align="center"><a href="../versionC/index.html" title="DKIT Lecture notes homepage for Derek O&#39; Reilly, Dundalk Institute of Technology (DKIT), Dundalk, County Louth, Ireland. Copyright Derek O&#39; Reilly, DKIT." target="_parent" style='font-size:0;color:white;background-color:white'>&nbsp;</a></div>