OPEN TO WORK

Available for new opportunities! Let's build something amazing together.

#DevHack: Dynamic manifests for Progressive Web App aka PWA

Development Devhack PWA
post

This post is over a year old, some of this information may be out of date.

In the previous #DevHack, I told you how to use deep links in your PWA for Microsoft Teams. This approach I currently implemented in one of my proof of concepts. Another functionality that my POC needed to have is to have dynamic manifest creating for the PWA.

{{< blockquote type="info" text="#DevHack: Deep linking to Microsoft Teams in Android and iOS

In this #DevHack, I will tell you how you can make it possible to get dynamic manifests for your PWA using Azure Static Web Apps and Azure Functions.

Approaches

When I started my journey with PWAs and dynamic manifests, I found two approaches:

  1. Create the manifest on the fly with JavaScript;
  2. Use a dynamically generated manifest.

The first approach is what I used from the start. This approach does not require anything special, except some extra JavaScript code to add the page’s manifest data.

This approach worked fine on iOS and Windows/macOS, but we noticed a different behavior when testing it on Android devices. Android never recognized our PWA as a “real” PWA. Instead, Android created it as a bookmark to the site.

Solution

The only working solution for dynamically generated manifests is to make use of an API. My POC created as an Azure Static Web App can handle this by adding an Azure Function.

import { AzureFunction, Context, HttpRequest } from "@azure/functions";
const httpTrigger: AzureFunction = async function (
context: Context,
req: HttpRequest
): Promise<void> {
const name = req.query.name;
const enityId = req.query.enityId;
const appId = req.query.appId;
context.res = {
body: {
name,
short_name: name,
description: name,
background_color: "#ffffff",
theme_color: "#ffffff",
display: "standalone",
start_url: `/entity/?appId=${appId}&enityId=${enityId}&name=${name}`,
icons: [
{
src: "/android-chrome-192x192.png",
sizes: "192x192",
type: "image/png",
},
{
src: "/android-chrome-512x512.png",
sizes: "512x512",
type: "image/png",
},
{
src: "/maskable_icon.png",
sizes: "196x196",
type: "image/png",
purpose: "any maskable",
},
],
},
headers: {
"Content-Type": "application/json",
},
};
};
export default httpTrigger;

In my React application, I call the Azure Function with the following logic:

<Helmet>
<link id="teams_manifest" rel="manifest"
href={`/api/manifest.webmanifest?${QS_APP_ID}=${encodeURIComponent(appId)}&${QS_ENTITY_ID}=${encodeURIComponent(enityId)}&${QS_NAME}=${appName}`}
/>
</Helmet>

Related articles

Report issues or make changes on GitHub

Found a typo or issue in this article? Visit the GitHub repository to make changes or submit a bug report.

Comments

Elio Struyf

Solutions Architect & Developer Expert

Loading...

Let's build together

Manage content in VS Code

Present from VS Code