Konubinix' opinionated web of thoughts

Simple Pwa Offline Camera

Fleeting

For the fun of it, I created a TWA using PWABuilder, available here (for the record, it also worked with bubblewrap).

open

{
    "name": "camera",
    "short_name": "camera",
    "description": "camera",
    "start_url": "./index.html",
    "display": "fullscreen",
    "background_color": "#ffffff",
    "theme_color": "#4285f4",
    "icons": [
        {
            "src": "./icons/icon-192x192.png",
            "type": "image/png",
            "sizes": "192x192"
        },
        {
            "src": "./icons/icon-512x512.png",
            "type": "image/png",
            "sizes": "512x512"
        }
    ],
    "orientation": "any",
    "prefer_related_applications": false,
    "scope": "./",
    "permissions": [],
    "splash_pages": null,
    "categories": []
}

const PRECACHE = 'precache-v1';
const RUNTIME = 'runtime';

const PRECACHE_URLS = [
    "index.html",
];


self.addEventListener('install', event => {
    event.waitUntil(
        caches.open(PRECACHE)
            .then(cache => cache.addAll(PRECACHE_URLS))
            .then(self.skipWaiting())
    );
});

self.addEventListener('activate', event => {
    const currentCaches = [PRECACHE, RUNTIME];
    event.waitUntil(
        caches.keys().then(cacheNames => {
            return cacheNames.filter(cacheName => !currentCaches.includes(cacheName));
        }).then(cachesToDelete => {
            return Promise.all(cachesToDelete.map(cacheToDelete => {
                return caches.delete(cacheToDelete);
            }));
        }).then(() => self.clients.claim())
    );
});

[[progressive_web_app.org:network_first-ex()]]

    <html>
      <head>
        <meta charset="UTF-8"/>
        <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=no" />
        <link rel="manifest" href="./manifest.json">
        <script defer src="https://konubinix.eu/ipfs/bafkreic33rowgvvugzgzajagkuwidfnit2dyqyn465iygfs67agsukk24i?orig=https://cdn.jsdelivr.net/npm/alpinejs@3.x.x/dist/cdn.min.js"></script>
        <script src="https://konubinix.eu/ipfs/bafybeihp5kzlgqt56dmy5l4z7kpymfc4kn3fnehrrtr7cid7cn7ra36yha?orig=https://cdn.tailwindcss.com/3.4.3"></script>
        <script>
window.onbeforeunload = function() {
        return "Dude, are you sure you want to leave? Think of the kittens!";
    }
          document.addEventListener('alpine:init', () => {
              Alpine.data('app', () => ({
                  video: null,
                  image_capture: null,
                  wakelock: null,
                  taking_pic: false,
                  algo: "canva",
                  async captureImage() {
                      this.taking_pic = true
                      if(this.algo === "take"){
                          blob = await this.image_capture.takePhoto()
                          // Create a link element to download the image
                          const link = document.createElement('a');
                          link.download = `photo_${this.getCurrentDateTime()}.png`;

                          const url = URL.createObjectURL(blob);
                          link.href = url;
                          link.click();
                          URL.revokeObjectURL(url);
                      }
                      else if(this.algo === "canva")
                      {
                          const canvas = document.createElement('canvas');
                          const ctx = canvas.getContext('2d');

                          canvas.width = video.videoWidth;
                          canvas.height = video.videoHeight;

                          ctx.drawImage(video, 0, 0, canvas.width, canvas.height);

                          // Convert canvas to blob
                          canvas.toBlob((blob) => {
                              // Create a link element to download the image
                              const link = document.createElement('a');
                              link.download = `photo_${this.getCurrentDateTime()}.png`;

                              const url = URL.createObjectURL(blob);
                              link.href = url;
                              link.click();
                              URL.revokeObjectURL(url);
                          }, 'image/png');
                      }
                      else if (this.algo === "grab")
                      {
                          imageBitmap = await this.image_capture.grabFrame()
                          const canvas = document.createElement('canvas');
                          canvas.width = imageBitmap.width;
                          canvas.height = imageBitmap.height;
                          const ctx = canvas.getContext('2d');
                          ctx.drawImage(imageBitmap, 0, 0);

                          canvas.toBlob(blob => {
                              // Create a link element to download the image
                              const link = document.createElement('a');
                              link.download = `photo_${this.getCurrentDateTime()}.png`;

                              const url = URL.createObjectURL(blob);
                              link.href = url;
                              link.click();
                              URL.revokeObjectURL(url);
                          }, 'image/png');

                      }
                      else{
                          alert(`unrecognize algo: ${this.algo}`)
                      }
                      setTimeout(() => {this.taking_pic = false}, 125)
                  },
                  getCurrentDateTime() {
                      const now = new Date();
                      const year = now.getFullYear();
                      const month = String(now.getMonth() + 1).padStart(2, '0');
                      const day = String(now.getDate()).padStart(2, '0');
                      const hours = String(now.getHours()).padStart(2, '0');
                      const minutes = String(now.getMinutes()).padStart(2, '0');
                      const seconds = String(now.getSeconds()).padStart(2, '0');
                      const milliseconds = String(now.getMilliseconds()).padStart(3, '0');

                      return `${year}-${month}-${day}_${hours}-${minutes}-${seconds}.${milliseconds}`;
                  },
                  async init () {
                      if ('serviceWorker' in navigator) {
                          navigator.serviceWorker
                              .register('./sw.js')
                              .then(function() {
                                  console.log('Service Worker Registered');
                              });
                      }
                      this.video = document.getElementById("video")
                      if(! 'mediaDevices' in navigator) {
                          alert( 'Your browser does not support media devices.' );
                      }
                      await this.setupMediaStream()
                      try{
                          navigator.wakeLock.request("screen").then(
                              (w) => {
                                  this.wakelock = w;
                                  this.message += "wakelock acquired"
                              }
                          ).catch((err) => {
                              this.message += `${err.name}, ${err.message}`;
                          });
                      } catch (err) {
                          this.message += err.toString()
                      };
                  },
                  async setupMediaStream() {
                      this.mediaStream = await navigator.mediaDevices.getUserMedia(
                          {
                              video: {
                                  facingMode: {
                                      exact: "environment"
                                  },
                              },
                              // audio: true,
                          }
                      );
                      const track = this.mediaStream.getVideoTracks()[0];
                      this.image_capture = new ImageCapture(track);
                      this.video.srcObject = this.mediaStream
                  },
              }))
          })
        </script>
      </head>
      <body x-data="app" class="text-slate-700" :class="{'bg-white': taking_pic, 'bg-black': !taking_pic}">
        <div>
          <div class="justify-center h-screen">
            <div class="relative h-screen">
              <div class="flex justify-center">
                <video id="video" @dblclick="$event.target.requestFullscreen()" class="max-w-screen h-screen" poster="https://d1tobl1u8ox4qn.cloudfront.net/2018/05/3814a83b-a2d4-4dfd-8945-f1cd003eb16f-1920x1080.jpg" autoplay="true" playsinline="true"></video>
              </div>
              <button class="absolute right-4 top-1/2" @click="captureImage"><img class="bg-white rounded-full w-10 h-10" src=""/></button>
            </div>
          </div>
        </div>
      </body>
    </html>

config

Notes linking here