Offline web application using beforeunload event

2019-08-24 02:27发布

问题:

Simply, I'm trying to make simple functionality for my web app when the browser being in the offline. The functionality is a simple message that tells the user about there is no network connectivity instead of the default's browser's page. I have tried the following:

window.addEventListener("beforeunload", function() {
                if (!navigator.onLine){
                    document.write("There is no network connection."); debugger;return false; 
                }                 
                }, false);

The above code worked with Chrome, but it does not work with firefox. I have two questions here:

  1. How could I make this cross-browsers?
  2. Is there any other way than using debugger?

Update

The answer of this question should be an implementation of service-worker

回答1:

Finally I found how to implement simple service worker for my website. The final result appears in the following screen shot:

After disconnecting the client's machine from the network, (Not use browser's development tools offline) and making the following steps, the offline.html is loaded for every website's page.

  1. I used an online tool that generates required service code, PWA Builder, I escaped the first step there, Generate Manifest, I started directly with step 2, Build Service Worker. I copied code for website from there, but I have modified some paths to point to the server's root because the code is going to be pasted in laravel application layout, like the following:

    // In the most top part of the layout HTML //This is the "Offline page" service worker

    //Add this below content to your HTML page, or add the js file to your page at the very top to register sercie worker
    if (navigator.serviceWorker.controller) {
      console.log('[PWA Builder] active service worker found, no need to register')
    } else {
      //Register the ServiceWorker
      navigator.serviceWorker.register('/pwabuilder-sw.js', { //notice slash in /pwabuilder-sw.js
        scope: './'
      }).then(function(reg) {
        console.log('Service worker has been registered for scope:'+ reg.scope);
      });
    }
    
    
    </script>
    

    2- I copied the code of Service worker code and saved it a new file in the web root of my application /public as pwabuilder-sw.js with also changing some paths with slashes:

    //This is the "Offline page" service worker

    //Install stage sets up the offline page in the cahche and opens a new cache
    self.addEventListener('install', function(event) {
      var offlinePage = new Request('/offline.html'); //notice /
      event.waitUntil(
      fetch(offlinePage).then(function(response) {
        return caches.open('pwabuilder-offline').then(function(cache) {
          console.log('[PWA Builder] Cached offline page during Install'+ response.url);
          return cache.put(offlinePage, response);
        });
      }));
    });
    
    //If any fetch fails, it will show the offline page.
    //Maybe this should be limited to HTML documents?
    self.addEventListener('fetch', function(event) {
      event.respondWith(
        fetch(event.request).catch(function(error) {
            console.error( '[PWA Builder] Network request Failed. Serving offline page ' + error );
            return caches.open('pwabuilder-offline').then(function(cache) {
              return cache.match('/offline.html'); //notice /
          });
        }));
    });
    
    //This is a event that can be fired from your page to tell the SW to update the offline page
    self.addEventListener('refreshOffline', function(response) {
      return caches.open('pwabuilder-offline').then(function(cache) {
        console.log('[PWA Builder] Offline page updated from refreshOffline event: '+ response.url);
        return cache.put(offlinePage, response);
      });
    });
    

Finally, I have created public/offline.html file with the following code:

<html>
 <head>
 <meta name="viewport" content="width=device-width, initial-scale=1">
  <title>OffLine</title>
  <style>
   body{
    text-align: center;
    padding: 1em;
    font-family: sans-serif;
    font-size: 30px;
    }
  </style>
 </head>
 <body>
  <img src="" alt="Disconnected" />
  <h1>OffLine</h1>
  <p>There is no network connectivity.</p>
  <a href="javascript:location.reload()">Click here<a/> to reload after connecting.
 </body>
</html>

Notice

For better results in testing this you may use the following hints:

  1. Clear your browser's cache and restart it.
  2. Disconnect the machine physically, ex: turn off the WiFi, After loading the website for the first time of course!

Additionally

I use jQuery to maintain some ajax requests all over the application, so in the layout script section, I used the following snippet that detect every ajax request before sending it and check if the browser is offline, so it stop the ajax request and alert a message then reload the page to load offline.html:

$( document ).ajaxSend(function(evt) {
    if (!navigator.onLine){
        evt.preventDefault();
        swal({
            title: '{{__('OffLine Error')}}',
            text: '{{__('The browser is offline. Some requests could not be done')}}',
            confirmButtonText: '<i class="fox-finalize"></i>{{__('OK')}}',
            onClose: function(){
                location.reload();
            }
        })
    }
  });