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>
      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