Creating a PWA
Introduction
PWA (Progressive Web Applications) offer several advantages that make them a desirable choice for web apps:
- Cross-platform compatibility: PWA can run on multiple platforms, including desktops, laptops, and mobile devices, without the need to develop separate native apps for each platform. 
- Installation: PWA can be installed directly from the browser, allowing users to access the app quickly from their home screen or app drawer without going through an app store. 
- Offline support: PWA can work offline or in poor network conditions by utilizing service workers and caching mechanisms. This ensures that users can still access and interact with the app even without an internet connection. 
- Push notifications: PWA can send push notifications to users, enabling real-time engagement and communication. This feature allows businesses to deliver important updates, promotions, or personalized messages to their users. 
- Improved user experience: PWA leverages web platform features to provide a seamless and engaging user experience. They can offer smooth animations, fast loading times, and a responsive design that adapts to different devices and screen sizes. 
- Cost-effective development: Developing a PWA can be more cost-effective compared to building separate native apps for different platforms. With a single codebase, developers can reach a broader audience and maintain the app more efficiently. 
By choosing a PWA approach, businesses and developers can provide their users with a native-like experience, increase user engagement, and reach a wider audience across various devices and platforms.
The following sections will explain how to transform your BF App into a PWA.
Getting Started
Requirements
- Running >V2.1.0 BF base code 
Making it installable
This will be the first feature added to your PWA. We will start by creating a manifest object and passing it as a parameter to a new BF function pwaSetManifest. In this example we added this code to a function in our onAppLoad named action.
let manifest = {
  name: "My Awesome PWA",
  short_name: "My App",
  icons: [{
      src: "LINK_TO_YOUR_ICON",
      sizes: "196x196",
      type: "image/png",
      purpose: "any"
    },{
      src: "LINK_TO_YOUR_ICON",
      sizes: "196x196",
      type: "image/png",
      purpose: "maskable"
    }],
  start_url: "https://your.domain.com/index.html",
  display: "standalone",
  background_color: "#000000",
  theme_color: "#4DBA87"
};
//background_color and theme_color can be customized
BF.pwaSetManifest(manifest);Next, as part of the onAppLoad as well, we add a new action pwaCustomInstall
{
    "action": "pwaCustomInstall",
    "options": {
        "beforeinstallprompt_actions": [{
            "action": "function",
            "function": "model.isInstalled = false;"
        }]
    }
}This new action has the options beforeinstallprompt, which are actions that will run before the trapped prompt to install event triggered by the browser when it identifies the web app as installable. The example above shows one way of setting a state variable that we will use to show or hide a button to install the app.
So in this case, we need a key isInstalled in our model.
In our example, once the install button is clicked on, a modal will prompt asking whether the user wants to install it.
{
    "actions": [{
        "action": "showModal",
        "options": {
            "body": "Do you want to get the most out of this app?",
            "buttons": [],
            "slots": [{
                "actions": [{
                    "action": "namedAction",
                    "name": "installApp"
                }, {
                    "action": "hideModal"
                }],
                "component": "button",
                "slot": "button",
                "styleClasses": "btn btn-danger",
                "text": "Install"
            }, {
                "actions": [{
                    "action": "hideModal"
                }],
                "component": "button",
                "slot": "button",
                "styleClasses": "btn btn-info",
                "text": "Not now"
            }]
        }
    }],
    "buttonClasses": "bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded",
    "styleClasses": "col-md-2",
    "text": "Install App!",
    "type": "button",
    "visible_calc": "!model.isInstalled"
}The named action installApp above has the following code.
{
    "action": "pwaPromptInstall",
    "options": {
        "onAccepted_actions": [{
            "action": "showAlert",
            "options": {
                "text": "Thanks for installing the app!",
                "title": "Welcome!",
                "type": "information"
            }
        }],
        "onDismissed_actions": [{
            "action": "showAlert",
            "options": {
                "text": "If I were you, I'd install the app!",
                "title": "Too bad!",
                "type": "information"
            }
        }]
    }
}The action pwaPromptInstall is the action that triggers the browser to prompt the user with the actual window that will install the app. This gives the flexibility to trigger this event from anywhere in your app, and customize it according to your needs.
The options onAccepted_actions and onDismissed_actions will run after the user click on Cancel, Install or close the browser’s prompt window.
Our modal looks like the image below.

By clicking on Install, it will trigger the browser event that asks if the user wants to install the app, as shown on the image below.

After installing it, the user will be able to access your web app as a regular native app.
Browser Support
(As of September 2022)
- Desktop and Laptop (reference: https://web.dev/learn/pwa/progressive-web-apps/#desktop-and-laptops) 
Windows 7
✔️ (73+)
✔️
❌
❌
Windows 8.x
✔️ (73+)
✔️
❌
❌
WIndows 10
✔️ (73+)
✔️( 79+)
✔️
❌
Windows 11
✔️ (73+)
✔️ (79+)
✔️
❌
Chrome OS
✔️ (72+)
❌
❌
✔️ (85+)
macOS
✔️ (73+)
✔️
❌
❌
Linux
✔️ (73+)
✔️
❌
❌
- Mobile Devices (reference: https://web.dev/learn/pwa/progressive-web-apps/#mobile-devices) - iOS and iPadOS: Safari (since iOS 11.3), App Store (since iOS/iPadOS 14, with some limitations), Mobile Configuration for enterprise distribution; 
- Android: Firefox, Google Chrome, Samsung Internet, Microsoft Edge, Opera, Brave, Huawei Browser, Baidu, UCWeb, Play Store (from version 72 with Google Chrome installed, or browsers compatible with TWA), Galaxy Store, Managed Play iframe for enterprise distribution. 
 
To support PWA behavior, you can use the checkPWAInstallation namedAction to detect whether the app is running in a browser or as an installed PWA. This can be useful for conditionally displaying elements like a PWA installation banner—show it only when the app is in the browser, and hide it when already installed. The action can be placed in the onFormLoad script to automatically update the state on load:
"checkPWAInstallation": [{
  "action": "function",
  "function": "const isStandalone = window.matchMedia('(display-mode: standalone)').matches || window.navigator.standalone;\n\nmodel.isPWAInstalled = isStandalone ? true : false;"
}]Sending Push Notifications
With PWA support, you can send push notifications to users, allowing your web app to engage with them. To request push notification permissions from users, we have an action that triggers the browser's 'ask for permission' prompt. This can be used at relevant and convenient points throughout your web app.
The example below shows how we could trigger the permission prompt from a button on a page.
{
    "actions": [{
        "action": "pwaPromptPushPermission",
        "options": {
            "idUser_calc": "window.vueapp.$store.state.auth.user.id"
        }
    }],
    "buttonClasses": "btn btn-info bg-blue-400",
    "styleClasses": "col-md-2",
    "text": "Prompt permission for push notifications",
    "type": "button"
}In this example, we set the user ID (BetterForms ID) as a key to target this when sending a push notification. Clicking on the button will display the following prompt.

Once the user clicks on Allow, the subscription will be saved in the BF cloud database and you will be able to send a push notification to the user using either an action from the frontend or hitting our /pushdata/sendnotification endpoint, as shown below.
Sending a Push Notification from an action:
Use the pwaPushNotificationSend action step.
// CD _ This is not most recents object shape
{
    "actions": [{
        "action": "pwaPushNotificationSend",
        "options": {
            "body": "Hello from BF push notification!",
            "data": {
                "dateOfArrival_calc": "Date.now()",
                "path": "/"
            },
            "filter": {
                "users": ["BF_USER_ID"]
            },
            "ttl": 300,
            "icon": "LINK_TO_YOUR_ICON",
            "title": "BF's New Feature!!!",
            "vibrate": [100, 50, 100]
        }
    }],
    "buttonClasses": "btn btn-info bg-blue-100",
    "styleClasses": "col-md-6",
    "text": "Send Push Notification",
    "type": "button"
}Sending a Push Notification from the API endpoint:
Hitting /pushdata/sendnotification endpoint ( Eg calling from FileMaker Server)
- Create a - POSTrequest to the following endpoint:- https://your.domain.com/pushdata/sendnotification
- Set the - bodyof the request will be as follows:
{
    "apiKey": "BFAPI_YOUR_KEY",
    "body": "Hello from BF push notification!",
    "data": {
        "dateOfArrival_calc": "Date.now()",
        "path": "/"
    },
    "icon": "LINK_TO_YOUR_ICON",
    "title": "BF's New Feature!!!",
    "vibrate": [100, 50, 100],
    "filter": {
        "users": ["BF_USER_ID"]
    },
    "ttl": 300
}Adding DOM Header Insertion to be available for offline use
- CSS files. The example below is loading tailwind version 2. 
<link href="https://unpkg.com/tailwindcss@^2/dist/tailwind.min.css" rel="stylesheet">
<script>
    function loadRemoteCSS(url) {
        $.get(url, function(cssContent) {
            $('<style>')
                .appendTo('head')
                .attr({
                    type: 'text/css'
                })
                .text(cssContent);
        });
    }
    $(document).ready(function() {
        loadRemoteCSS('https://unpkg.com/tailwindcss@^2/dist/tailwind.min.css');
    });
</script>Last updated
