#datauri_info
data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>datauri info</title>
    <style>
      body {
        margin: 1em 1em 3em 1em;
      }

      header h1,
      header h2 {
        display: inline-block;
        margin-right: 1em;
      }

      .menu,
      .content {
        display: none;
      }

      .active {
        display: block;
      }

      h2 > a {
        padding: .2em .3em;
      }

      a:visited {
        color: inherit;
      }

      a:hover {
        color: rgb(200, 0, 0);
      }

      .data {
        background: rgb(220, 220, 100);
        /*background: rgb(180, 200, 230);*/
        /*background: rgb(180, 200, 230);*/
        overflow: hidden;
        word-break: break-all;
        max-height: 10em;
        padding: .3em;
      }

      .data,
      a {
        border-radius: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

      h2 {
        margin: 1em 0 .4em 0;
      }

      h2:first-child {
        margin-top: 0;
      }

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <div class="menu active">
        <h1>datauri info</h1>
        <h2>
          <a href="#devs">datauri for devs</a>
        </h2>
      </div>
      <div class="menu">
        <h1>datauri for devs</h1>
        <h2>
          <a href="">datauri info</a>
        </h2>
      </div>
    </header>
    <main>
      <div class="content active">
        <p>
        What is datauri application? Apps hosted as datauris are basically portable web apps that work without http server.
        If you see this page, you probably already know how datauri apps work. You just copy-paste the link and access the app.
        Here are some datauri apps.
        </p>
        <h2>wired</h2>
        <p>Anon approved pow based nostr client</p>
        <p class="data">data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K</p>

        <h2>ourchan</h2>
        <p>Anon approved image board</p>
        <p class="data">data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>filepublish</h2>
        <p>Upload files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>fileloader</h2>
        <p>Download/view files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>video clip creator</h2>
        <p>Create and upload 1 second video clips</p>
        <p class="data">data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <title>video clip creator</title>
    <style>
    body {
      margin: 1em;
    }
    h1 {
      margin-top: 0;
      font-size: 1.5em;
    }
    #log {
      overflow: auto;
      position: absolute;
      bottom: 1em;
      top: 11em;
      border: .3em solid #666;
      border-radius: .3em;
      right: 1em;
      left: 1em;
      padding: .5em;
      background: #eee;
    }
    #progress {
      background: #ddd;
      border-radius: .3em;
    }
    #progressbar {
      height: 1.3em;
      background: blue;
      width: 0%;
      border-radius: .4em;
    }
    #log > a {
      margin-bottom: 1em;
      display: block;
    }
    p {
      margin-top: 0;
      max-height: 7em;
      overflow: scroll;
    }
    video {
      display: none;
      z-index: 1;
    }
    #canvas,
    video {
      position: absolute;
      top: 1em;
      right: 1em;
      width: 10em;
      height: 6em;
    }
    </style>
  </head>
  <body>
    <h1>video clip creator</h1>
    <form id="form">
      <p>
        <input id="file" type="file"/>
      </p>
      <p>
        <input id="submit" type="submit" value="encode"/>
      </p>
    </form>
    <video id="video"></video>
    <video id="video2" controls loop="1"></video>
    <canvas id="canvas"></canvas>
    <div id="progress">
      <div id="progressbar"></div>
    </div>
    <div id="log"><div>
    <script type="module">
    const canvas = document.getElementById("canvas")
    const ctx = canvas.getContext('2d')
    const log = document.getElementById("log")
    const video = document.getElementById("video")
    const video2 = document.getElementById("video2")
    const progressbar = document.getElementById("progressbar")
    let files = null

    const frame_duration = 1/30
    const extract_frame_count = 30//1 / frame_duration
    let time = 15

    const buffer = await fetch(
      "https://unpkg.com/webm-wasm@0.4.1/dist/webm-worker.js"
    ).then(r => r.arrayBuffer())

    const worker = new Worker(
      URL.createObjectURL(new Blob([buffer], { type: "text/javascript" }))
    )

    let encoded_data = []
    let start_time = null
    let ready = false

    document.getElementById("form").onsubmit = function(e){
      e.preventDefault()
      video2.style.display = "none";
      video2.pause()
      progressbar.style.width = "0.5%"
      this.querySelectorAll("input").forEach(el => el.disabled = true)
      document.getElementById("submit").value = "encoding"
      files = document.getElementById("file").files
      start()
    }

    function blobToDataURL(blob, callback) {
      return new Promise(resolve => {
        var a = new FileReader()
        a.onload = function(e) {
          resolve(e.target.result)
        }
        a.readAsDataURL(blob)
      })
    }

    function info(msg, type){
      const el = document.createElement(type === 2 && "a" || "p")
      el[(type === 3) && "innerHTML" || "innerText"] = msg

      if(type){
        el.href = msg
      }

      log.append(el)
      log.scrollTo(0, 1e6)
    }

    async function process_video(){
      let frame_count = 0

      video.onseeked = function(e) {
        frame_count++
        info("extract frame " + frame_count + " / " + extract_frame_count)

        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height)

        if(frame_count < extract_frame_count){
          time += frame_duration
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          worker.postMessage(imageData.data.buffer, [imageData.data.buffer])
        }else{
          worker.postMessage(null)
        }
      }
    }

    function total_time(start){
      return (new Date().getTime() - start) / 1000
    }

    function createBufferURL(buffer, type = '') {
      return URL.createObjectURL(new Blob([buffer], {type}));
    }

    async function show_data_uri(blob){
      let blob_url = URL.createObjectURL(blob)

      info(await blobToDataURL(blob))
      info(blob_url, 2)

      video2.src = blob_url
      video2.style.display = "block";
      video2.play()

      progressbar.style.width = "100%"
      document.getElementById("form").reset()
      document.getElementById("submit").value = "encode"
      document.querySelectorAll("form input").forEach(el => el.disabled = false)
    }

    worker.onmessage = function(e){
      if(!e.data) {
        return
      }

      if(e.data != null && typeof e.data == "object" && e.data.byteLength > 0){
        encoded_data.push(e.data)
        info("encode " + encoded_data.length + " / " + extract_frame_count)
        progressbar.style.width = parseInt((encoded_data.length / extract_frame_count) * 100, 10) + "%"
      }

      if(e.data == "READY"){
        info("ready", canvas.width, canvas.height)

        worker.postMessage({
          width: canvas.width,
          height: canvas.height,
          realtime: true,
          bitrate: 100
        })

        //for(let i = 0; i < frames.length; i++){
        //  worker.postMessage(frames[i], [frames[i]])
        //}

        ready = true
        process_video()
      }
      else if(ready){
        if(e.data.byteLength == 0){
          info("encoding finished (duration: " + parseInt(total_time(start_time), 10) +
            " s, speed: " + parseInt(total_time(start_time) / extract_frame_count, 10) +" s / frame)")
          let blob = new Blob(encoded_data, { type: 'video/webm' })
          show_data_uri(blob)
        }
      }
    }

    function load_video(){
      return new Promise(resolve => {
        video.src = URL.createObjectURL(files[0])

        video.onloadedmetadata = function() {
          canvas.height = video.videoHeight
          canvas.width = video.videoWidth
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          resolve()
        }
      })
    }

    async function start(){
      start_time = new Date().getTime()
      window.start_time = start_time
      await load_video()
      worker.postMessage("https://unpkg.com/webm-wasm@0.4.1/dist/webm-wasm.wasm")
    }
    </script>
  </body>
</html>
</p>
      </div>
      <div class="content">
        <h2>truly distributed, local first apps</h2>
        <p>
        Datauri apps dont rely on http server for initialization. Many apps however load dynamic content from internet.
        Releasing datauri apps is very simple, because there is no need for web server or domain name.
        </p>
        <p>Using nostr as backbone for dynamic content provides distributed alternative for centralized solutions.
        It is simple to load content from nostr notes. Nostr is based on websocket connection. You can publish ~50 kB data on
        one nostr note, which is just json wrapper for any data content.
        See <a href="https://www.e2encrypted.com/nostr/nips/">nostr documentation</a>
        for more information.</p>

        <h2>immutable or updatable content</h2>
        <p>
        Datauris are by default immutable, ie. developer cannot update its content after its published,
        however this is not true for dynamically loaded content. Developer can use dynamic content loader
        to load full page source from nostr event. Below is an example.
        </p>
        <ul>
          <li>r = relay_uri</li>
          <li>k = pubkey</li>
          <li>f = d_tag</li>
        </ul>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>
        <p>In this case, developer can publish new nostr event using same pubkey and same d tag value, which will override the previous content</p>
      </div>
    </main>
    <script>
    document.onclick = function(e){
      if(!e.target.classList.contains("data")){
        for(let datauri of document.querySelectorAll("p.data")){
          datauri.classList.remove("scroll")
        }
      }
    }

    for(let datauri of document.querySelectorAll("p.data")){
      datauri.onclick = function(){
        const range = document.createRange()
        range.selectNode(this)
        getSelection().removeAllRanges()
        getSelection().addRange(range)
        this.classList.add("scroll")
      }
    }

    function apply_hash(){
      const active_index = location.hash === "#devs" ? 1 : 0
      document.querySelector(".menu.active").classList.remove("active")
      document.querySelector(".content.active").classList.remove("active")
      document.querySelectorAll(".menu")[active_index].classList.add("active")
      document.querySelectorAll(".content")[active_index].classList.add("active")
    }

    addEventListener("hashchange", apply_hash)
    apply_hash()
    </script>
  </body>
</html>
