Avatar
b
87ae535006d142ab4ffb28c964241bca3a2f98801f5ce6282cd97f8b8d6b32ed

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


#datauri #info

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

      .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;
        border-radius: .3em;
        padding: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

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

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <h1>datauri info</h1>
    </header>
    <main>
      <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>
    </main>
    <footer></footer>
    <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")
      }
    //  datauri.onwheel = function(e){
    //    //e.preventDefault()
    //    console.log("wheel")
    //  }
    }
    </script>
  </body>
</html>


#wired #wiredapp #standaloneapp updated source code for building datauri compatible release

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#0459bf5c4ce01f95e2a0ca581e738f1365aaf39d328c33d661a6171794f494f9/inline/quiet

here is direct link for #filepublish

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K

bootstrap wiredapp from datauri

#wired #wiredapp #p2p #decentralized #datauri #stadaloneapp #serverless #hostr #nostr #pow

data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K

#wired #wiredapp standalone version source code here

wip

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#5a89379d7a65f16fc4034a60f77086d3068dafe77e71aeea28cf533fa510cd7e/inline/quiet

#wired #wiredapp

wip

there webworkers are bit difficult (posting does not work)

http://bafkreigjwik74seutskr6mbibubphu26vhhrbz4d3lyffn5eeymtldd2ci.ipfs.cf-ipfs.com/

Replying to Avatar b

dget_example.html

dget is super simple way to load content from nostr

here is simple example

upload two images, ~2MB each:

node ~/filepublish.mjs pic1.png -Q

node ~/filepublish.mjs pic2.png -Q

fetch images:

await dget("id1", "wss://nos.lol")

await dget("id2", "wss://nos.lol")

full html example:

https://ipfs.io/ipfs/QmZahCwYgGmfU1mMnGCNWyivZaiqt6iL12fLtm8s8eg4WP

full html example source:

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#8ed91c8dd15e236b896231e84d20263c55bfc2d130d358f5284ac3b38cf654f0/inline/quiet

#dget #dget_example

embedding image is simple as this:

dget_example.html

dget is super simple way to load content from nostr

here is simple example

upload two images, ~2MB each:

node ~/filepublish.mjs pic1.png -Q

node ~/filepublish.mjs pic2.png -Q

fetch images:

await dget("id1", "wss://nos.lol")

await dget("id2", "wss://nos.lol")

full html example:

https://ipfs.io/ipfs/QmZahCwYgGmfU1mMnGCNWyivZaiqt6iL12fLtm8s8eg4WP

full html example source:

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#8ed91c8dd15e236b896231e84d20263c55bfc2d130d358f5284ac3b38cf654f0/inline/quiet

#dget #dget_example

find . -type f | while read f; do newfile=$(node ~/filepublish.mjs "$f" -Q); echo "/${f:2}|$newfile"; done

/512_tao.png|7ccc02f1666384be5fc505d96c18a9a720ab8e92352ff42fcd8b33c6f9680e8e

/favicon.gif|3a996c7bd79720f727a82ace8db1304bebac0ac5ec3d64ed9bfdcaa1651789bf

...

await dget("7ccc02f1666384be5fc505d96c18a9a720ab8e92352ff42fcd8b33", "wss://nos.lol")

#dget #dgetjs #dgethtml

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#bca68da18eb94dcf0d064b4f91f046e9d6b7366d36023dcd1b350eab55193984/inline/quiet

#filepublishmjs #filepublishnode #filepublish

upload:

node filepublish.mjs hello.txt

in browser js app, get blob uri:

await dget("id", "relay")

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#759233efbf69c7961295a27b157afdf02bb6234a61f718d6b11b6c679e3698a8/inline/quiet

#filepublishmjs #filepublishnode #filepublish

node filepublish.mjs hello.txt

data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#c5737824ae1c74c8bd11c28fcb4c5822be65a78bf57cccfdc215c75165b6404a/inline/quiet

Replying to Avatar b

so here are steps for doing #decentralized #web #apps

if you want to use react (because your boss told you its necessary)

- use hashrouter https://reactrouter.com/en/main/router-components/hash-router

- make datauri, like this:

"data:text/html;base64,"$(cat index.html | base64 -w 0)

your app can now be shared without http server

when you need more content, upload it to relays. for small files less than 100 kB #hostr or other simple way of using event content is ok. if there is need for larger files, #filepublish #fileloader #dget gets the job done.

i highly recommend using vanilla js, as your codebase can be 1/100th of the size compared to react app.

for bootstrapping content from #hostr with short #datauri, you can use hostr loader like this:

r = relay

k = pubkey

f = d_tag

rest of the #code is #javascript code for loading content from #nostr event

data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K

so here are steps for doing #decentralized #web #apps

if you want to use react (because your boss told you its necessary)

- use hashrouter https://reactrouter.com/en/main/router-components/hash-router

- make datauri, like this:

"data:text/html;base64,"$(cat index.html | base64 -w 0)

your app can now be shared without http server

when you need more content, upload it to relays. for small files less than 100 kB #hostr or other simple way of using event content is ok. if there is need for larger files, #filepublish #fileloader #dget gets the job done.

i highly recommend using vanilla js, as your codebase can be 1/100th of the size compared to react app.

its funny how web degenerated

>invent html markup language

>now everyone can make websites

>make js

>make google, facebook, youtube

>make react

>make all websites using react

>get 100x performance degradation

>fill all websites with ads

>fill all websites with cookie popups

>now everyone uses only 3 website for all their web surfing needs

could as well have downloaded 3 native apps instead of even inventing html and js

we could be running pentium 3 machines and get same performance today, websites were built with vanillajs rather than react which adds 100x overhead in filesize and probably 10x overhead in performance