Avatar
dev
ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87

https://ipfs.io/ipfs/QmXDLNkXQ5vSjZ8N5jtN5jjqt3jmKNGPkgTDe99ocAW5rr

files.html, fix: user profile names

sha256

6db875c03a1f79a7203b8cbf20700841bb2cb0e95c020a11c54ab239d1d63e75 files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs, 
      #privkey {
        display: block;
        width: 18em;
        background: #111;
        color: #fff;
      }
      
      #relays,
      #logs {
        height: 6em;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      select {
        width: 4em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.5em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="1" href="#">1</a>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">private key:</div>
        <p>If you cant use browser addon (mobile)</p>
        <form action="#" id="privkey-form">
          <input type="text" id="privkey"/>
          <input class="submit" type="submit" value="save key"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js"></script>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")
      qs("#privkey").value = privkey()
      const colopts = [1, 2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)

      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let likes = []
      let newlikes = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let querycount = 0
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function privkey(){
        let key = localStorage.getItem("privkey") || ""
        
        if(key.length == 63){
          key = NostrTools.nip19.decode(key).data
        }
        
        return key.length == 64 && key || null
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
      })

      qs("#privkey-form").addEventListener("submit", async e => {
        e.preventDefault()
        const key = qs("#privkey").value

        if(key.length != 64 && key.length != 63){
          alert("key should be 63 or 64 characters long")
          return
        }
        
        localStorage.setItem("privkey", key)
        alert("key saved")
      })
      
      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      function haskey(){
        return window.nostr || privkey()
      }
      
      async function pubkey(){
        return window.nostr && await window.nostr.getPublicKey() || NostrTools.getPublicKey(privkey())
      }
      
      async function sign(event){
        if(window.nostr){
          return await nostr.signEvent(event)
        }
        
        event.sig = NostrTools.getSignature(event, privkey())
        return event
      }
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubk = await pubkey()
        const like = await sign({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubk,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        const likejson = JSON.stringify(["EVENT", like])
        sockets.forEach(s => s.send(likejson))
        console.log("sent like", likejson) 
        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(res, socket, q, type, stime){
        save(socket, res, q, type, false, false)
        log("query res", res.length, socket.url)
        querycount--

        if(querycount == 0){
          qs("#loading").style.display = "none"
        }
        
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        let qidlikes = "likes_" + (++idcounter)

        if(newlikes.length > 0){
          primalsocket.send(JSON.stringify([
            "REQ", 
            qidlikes, 
            {
              "cache": [
                "events", {
                  "event_ids": newlikes
                }
              ]
            }
          ]))

          const like_events = await waitres(qidlikes)
          newlikes.splice(0)
          
          await save(socket, like_events, "", "all", true, true)
        }
        
        socket.send(JSON.stringify([
          "REQ", 
          qid, 
          {
            "cache": [
              "explore", 
              {
                "timeframe": "latest",
                "scope": "global",
                "limit": 20
              }
            ]
          }
        ]))
        
        const res = await waitres(qid)
        await handlequeryres(res.filter(e => e.kind == 1), socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        
        const likesres = await querylikes(socket)
        
        likesres.forEach(eventid => {
          if(!likes.includes(eventid)){
            likes.push(eventid)
            newlikes.push(eventid)
          }
        })
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)
        socket.send(searchjson)
        
        const res = await waitres(qid)
        await handlequeryres(res, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket){
        if(!haskey()){
          return []
        }
        
        const mypubkey = await pubkey()
        const events = await queryevents(socket, [7], null, [mypubkey])
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])

        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubk){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubk))

        return (await waitres(queryid))
      }
      
      async function save(socket, eventdb, q, qtype, reacted, append){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }

          return true
        })
        
        let results = []
        const eventids = events.map(e => e.id)
        let newurls = []
        
        for(let event of events){
          let urlsmatch = []
          qtype != "video" && urlsmatch.push(...(event.content.match(imgregex) || []))
          qtype != "image" && urlsmatch.push(...(event.content.match(videoregex) || []))
          urlsmatch = urlsmatch.filter(url => !urls.includes(url))

          if(urlsmatch.length > 0){
            newurls.push({
              event: event,
              urls: urlsmatch
            })
            
            urls.push(...urlsmatch)
            eventpubkeys.includes(event.pubkey) || eventpubkeys.push(event.pubkey)
          }
        }
        
        log("query meta", socket.url)
        let metaevents = await queryevents(socket, [0], null, eventpubkeys)
        
        for(let entry of newurls){
          for(let url of entry.urls){
            results.push({
              reacted: reacted || likes.includes(entry.event.id), 
              url: url, 
              event: entry.event, 
              bech32id: bech32.encode("note", fromHexString(entry.event.id)),
              meta: metaevents.find(o => o.pubkey == entry.event.pubkey)
            })
          }
        }
        
        log("save res", results.length, socket.url)
        updatecontent(results, q, append)
      }
      
      function updatecontent(results, q, append){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics")[append && "append" || "prepend"](root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        const pic = gid("pic")

        if(item.url.match(videoextregex)){
          pic.style.display = "none"
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          pic.src = item.url
          pic.style.display = "block"
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(haskey()){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmZMyWmLsSoU87vtqjPaVoNukwJ83S2NBs98fSDAgKrTym

files.html, add single column option

sha256

942ff547df74448d52fb45a5094dff0977098f3835b70cbbf1c8dbd4e0950367 files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs, 
      #privkey {
        display: block;
        width: 18em;
        background: #111;
        color: #fff;
      }
      
      #relays,
      #logs {
        height: 6em;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      select {
        width: 4em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.5em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="1" href="#">1</a>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">private key:</div>
        <p>If you cant use browser addon (mobile)</p>
        <form action="#" id="privkey-form">
          <input type="text" id="privkey"/>
          <input class="submit" type="submit" value="save key"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js"></script>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")
      qs("#privkey").value = privkey()
      const colopts = [1, 2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)

      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let likes = []
      let newlikes = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let querycount = 0
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function privkey(){
        let key = localStorage.getItem("privkey") || ""
        
        if(key.length == 63){
          key = NostrTools.nip19.decode(key).data
        }
        
        return key.length == 64 && key || null
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
      })

      qs("#privkey-form").addEventListener("submit", async e => {
        e.preventDefault()
        const key = qs("#privkey").value

        if(key.length != 64 && key.length != 63){
          alert("key should be 63 or 64 characters long")
          return
        }
        
        localStorage.setItem("privkey", key)
        alert("key saved")
      })
      
      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      function haskey(){
        return window.nostr || privkey()
      }
      
      async function pubkey(){
        return window.nostr && await window.nostr.getPublicKey() || NostrTools.getPublicKey(privkey())
      }
      
      async function sign(event){
        if(window.nostr){
          return await nostr.signEvent(event)
        }
        
        event.sig = NostrTools.getSignature(event, privkey())
        return event
      }
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubk = await pubkey()
        const like = await sign({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubk,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        const likejson = JSON.stringify(["EVENT", like])
        sockets.forEach(s => s.send(likejson))
        console.log("sent like", likejson) 
        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(res, socket, q, type, stime){
        save(socket, res, q, type, false, false)
        log("query res", res.length, socket.url)
        querycount--

        if(querycount == 0){
          qs("#loading").style.display = "none"
        }
        
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        let qidlikes = "likes_" + (++idcounter)

        if(newlikes.length > 0){
          primalsocket.send(JSON.stringify([
            "REQ", 
            qidlikes, 
            {
              "cache": [
                "events", {
                  "event_ids": newlikes
                }
              ]
            }
          ]))

          const like_events = await waitres(qidlikes)
          newlikes.splice(0)
          
          await save(socket, like_events, "", "all", true, true)
        }
        
        socket.send(JSON.stringify([
          "REQ", 
          qid, 
          {
            "cache": [
              "explore", 
              {
                "timeframe": "latest",
                "scope": "global",
                "limit": 20
              }
            ]
          }
        ]))
        
        const res = await waitres(qid)
        await handlequeryres(res.filter(e => e.kind == 1), socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        
        const likesres = await querylikes(socket)
        
        likesres.forEach(eventid => {
          if(!likes.includes(eventid)){
            likes.push(eventid)
            newlikes.push(eventid)
          }
        })
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)
        socket.send(searchjson)
        
        const res = await waitres(qid)
        await handlequeryres(res, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket){
        if(!haskey()){
          return []
        }
        
        const mypubkey = await pubkey()
        const events = await queryevents(socket, [7], null, [mypubkey])
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])

        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubk){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubk))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reacted){
        let results = []

        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)

            urls.push(url)
            results.push({
              reacted: reacted || likes.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype, reacted, append){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }

          eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
          return true
        })
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []

        log("query meta", socket.url)
        
        //metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reacted))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reacted))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q, append)
      }
      
      function updatecontent(results, q, append){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics")[append && "append" || "prepend"](root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        const pic = gid("pic")

        if(item.url.match(videoextregex)){
          pic.style.display = "none"
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          pic.src = item.url
          pic.style.display = "block"
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(haskey()){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmePrZ26tpriwdt9RnFjRa79LaewwLCpG3QZkMExky7ufc

files.html, fix add like

sha256

d489b50d426f3dd9f81ee7cb50daa45c73984de3425d3148a5ee99020f12e54d files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs, 
      #privkey {
        display: block;
        width: 18em;
        background: #111;
        color: #fff;
      }
      
      #relays,
      #logs {
        height: 6em;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      select {
        width: 4em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.5em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">private key:</div>
        <p>If you cant use browser addon (mobile)</p>
        <form action="#" id="privkey-form">
          <input type="text" id="privkey"/>
          <input class="submit" type="submit" value="save key"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js"></script>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")
      qs("#privkey").value = privkey()
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)

      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let likes = []
      let newlikes = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let querycount = 0
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function privkey(){
        let key = localStorage.getItem("privkey") || ""
        
        if(key.length == 63){
          key = NostrTools.nip19.decode(key).data
        }
        
        return key.length == 64 && key || null
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
      })

      qs("#privkey-form").addEventListener("submit", async e => {
        e.preventDefault()
        const key = qs("#privkey").value

        if(key.length != 64 && key.length != 63){
          alert("key should be 63 or 64 characters long")
          return
        }
        
        localStorage.setItem("privkey", key)
        alert("key saved")
      })
      
      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      function haskey(){
        return window.nostr || privkey()
      }
      
      async function pubkey(){
        return window.nostr && await window.nostr.getPublicKey() || NostrTools.getPublicKey(privkey())
      }
      
      async function sign(event){
        if(window.nostr){
          return await nostr.signEvent(event)
        }
        
        event.sig = NostrTools.getSignature(event, privkey())
        return event
      }
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubk = await pubkey()
        const like = await sign({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubk,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        const likejson = JSON.stringify(["EVENT", like])
        sockets.forEach(s => s.send(likejson))
        console.log("sent like", likejson) 
        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(res, socket, q, type, stime){
        save(socket, res, q, type, false, false)
        log("query res", res.length, socket.url)
        querycount--

        if(querycount == 0){
          qs("#loading").style.display = "none"
        }
        
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        let qidlikes = "likes_" + (++idcounter)

        if(newlikes.length > 0){
          primalsocket.send(JSON.stringify([
            "REQ", 
            qidlikes, 
            {
              "cache": [
                "events", {
                  "event_ids": newlikes
                }
              ]
            }
          ]))

          const like_events = await waitres(qidlikes)
          newlikes.splice(0)
          
          await save(socket, like_events, "", "all", true, true)
        }
        
        socket.send(JSON.stringify([
          "REQ", 
          qid, 
          {
            "cache": [
              "explore", 
              {
                "timeframe": "latest",
                "scope": "global",
                "limit": 20
              }
            ]
          }
        ]))
        
        const res = await waitres(qid)
        await handlequeryres(res.filter(e => e.kind == 1), socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        
        const likesres = await querylikes(socket)
        
        likesres.forEach(eventid => {
          if(!likes.includes(eventid)){
            likes.push(eventid)
            newlikes.push(eventid)
          }
        })
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)
        socket.send(searchjson)
        
        const res = await waitres(qid)
        await handlequeryres(res, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket){
        if(!haskey()){
          return []
        }
        
        const mypubkey = await pubkey()
        const events = await queryevents(socket, [7], null, [mypubkey])
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])

        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubk){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubk))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reacted){
        let results = []

        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)

            urls.push(url)
            results.push({
              reacted: reacted || likes.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype, reacted, append){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }

          eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
          return true
        })
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []

        log("query meta", socket.url)
        
        //metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reacted))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reacted))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q, append)
      }
      
      function updatecontent(results, q, append){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics")[append && "append" || "prepend"](root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        const pic = gid("pic")

        if(item.url.match(videoextregex)){
          pic.style.display = "none"
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          pic.src = item.url
          pic.style.display = "block"
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(haskey()){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmfBxzFp3VZEZD3V3TYxWWbkeBc53J8n4XpbLcBekTTMLo

files.html, add privkey option for mobile users (likes view on mobile)

sha256

b78d2dcedaef1c59a644cad675a7bc0b7850aa20f50ad9769f49c850060033a8 files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs, 
      #privkey {
        display: block;
        width: 18em;
        background: #111;
        color: #fff;
      }
      
      #relays,
      #logs {
        height: 6em;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      select {
        width: 4em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.5em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">private key:</div>
        <p>If you cant use browser addon (mobile)</p>
        <form action="#" id="privkey-form">
          <input type="text" id="privkey"/>
          <input class="submit" type="submit" value="save key"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js"></script>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")
      qs("#privkey").value = privkey()
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)

      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let likes = []
      let newlikes = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let querycount = 0
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function privkey(){
        let key = localStorage.getItem("privkey") || ""
        
        if(key.length == 63){
          key = NostrTools.nip19.decode(key).data
        }
        
        return key.length == 64 && key || null
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
      })

      qs("#privkey-form").addEventListener("submit", async e => {
        e.preventDefault()
        const key = qs("#privkey").value

        if(key.length != 64 && key.length != 63){
          alert("key should be 63 or 64 characters long")
          return
        }
        
        localStorage.setItem("privkey", key)
        alert("key saved")
      })
      
      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      function haskey(){
        return window.nostr || privkey()
      }
      
      async function pubkey(){
        return window.nostr && await window.nostr.getPublicKey() || NostrTools.getPublicKey(privkey())
      }
      
      async function sign(event){
        if(window.nostr){
          return await nostr.signEvent(event)
        }
        
        event.sig = NostrTools.getSignature(event, privkey())
        return event
      }
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await pubkey()
        const like = await sign({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(res, socket, q, type, stime){
        save(socket, res, q, type, false, false)
        log("query res", res.length, socket.url)
        querycount--

        if(querycount == 0){
          qs("#loading").style.display = "none"
        }
        
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        let qidlikes = "likes_" + (++idcounter)

        if(newlikes.length > 0){
          primalsocket.send(JSON.stringify([
            "REQ", 
            qidlikes, 
            {
              "cache": [
                "events", {
                  "event_ids": newlikes
                }
              ]
            }
          ]))

          const like_events = await waitres(qidlikes)
          newlikes.splice(0)
          
          await save(socket, like_events, "", "all", true, true)
        }
        
        socket.send(JSON.stringify([
          "REQ", 
          qid, 
          {
            "cache": [
              "explore", 
              {
                "timeframe": "latest",
                "scope": "global",
                "limit": 20
              }
            ]
          }
        ]))
        
        const res = await waitres(qid)
        await handlequeryres(res.filter(e => e.kind == 1), socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        
        const likesres = await querylikes(socket)
        
        likesres.forEach(eventid => {
          if(!likes.includes(eventid)){
            likes.push(eventid)
            newlikes.push(eventid)
          }
        })
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)
        socket.send(searchjson)
        
        const res = await waitres(qid)
        await handlequeryres(res, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket){
        if(!haskey()){
          return []
        }
        
        const mypubkey = await pubkey()
        const events = await queryevents(socket, [7], null, [mypubkey])
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])

        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reacted){
        let results = []

        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)

            urls.push(url)
            results.push({
              reacted: reacted, 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype, reacted, append){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }

          eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
          return true
        })
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []

        log("query meta", socket.url)
        
        //metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reacted))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reacted))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q, append)
      }
      
      function updatecontent(results, q, append){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics")[append && "append" || "prepend"](root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        const pic = gid("pic")

        if(item.url.match(videoextregex)){
          pic.style.display = "none"
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          pic.src = item.url
          pic.style.display = "block"
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(haskey()){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmeM9KFuuFmoE9LsN6PqKVtbdhFCL95v3QEdXKqpdgAy5m

files.html, improved likes view

sha256

fd3ffba18eb6f6330094b00cebf3faaf097f84928bba79948b3eb009cb9874e9 files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let likes = []
      let newlikes = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let querycount = 0
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(res, socket, q, type, stime){
        save(socket, res, q, type, false, false)
        log("query res", res.length, socket.url)
        querycount--

        if(querycount == 0){
          qs("#loading").style.display = "none"
        }
        
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        let qidlikes = "likes_" + (++idcounter)

        if(newlikes.length > 0){
          primalsocket.send(JSON.stringify([
            "REQ", 
            qidlikes, 
            {
              "cache": [
                "events", {
                  "event_ids": newlikes
                }
              ]
            }
          ]))

          const like_events = await waitres(qidlikes)
          newlikes.splice(0)
          
          await save(socket, like_events, "", "all", true, true)
        }
        
        socket.send(JSON.stringify([
          "REQ", 
          qid, 
          {
            "cache": [
              "explore", 
              {
                "timeframe": "latest",
                "scope": "global",
                "limit": 20
              }
            ]
          }
        ]))
        
        const res = await waitres(qid)
        await handlequeryres(res.filter(e => e.kind == 1), socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        querycount++
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        
        const likesres = await querylikes(socket)
        
        likesres.forEach(eventid => {
          if(!likes.includes(eventid)){
            likes.push(eventid)
            newlikes.push(eventid)
          }
        })
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)
        socket.send(searchjson)
        
        const res = await waitres(qid)
        await handlequeryres(res, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], null, [mypubkey])
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])

        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reacted){
        let results = []

        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)

            urls.push(url)
            results.push({
              reacted: reacted, 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype, reacted, append){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }

          eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
          return true
        })
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []

        log("query meta", socket.url)
        
        //metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reacted))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reacted))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q, append)
      }
      
      function updatecontent(results, q, append){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics")[append && "append" || "prepend"](root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmYYVVMmtMGrrj48BjkZCznxbeCDFHLsY3hvESkB6strpw

files.html, add primal global feed back

sha256

e51c258fcbf7c652b09b70257f097591a3bb3ee80181e8fa05c3ee1aefc40080 files.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">connections:</div>
        <div id="conncount">0</div>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save relays"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let primalsocket = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function handlequeryres(qid, socket, q, type, stime){
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - stime)
        querywait = sleep(wait)
        await querywait.promise
      }
      
      async function primalglobal(socket){
        log("global", socket.url)
        let stime = Date.now()
        let qid = "global_" + (++idcounter)
        
        socket.send(JSON.stringify(["REQ", qid, {
          "cache": [
            "explore", {
              "timeframe": "latest",
              "scope": "global",
              "limit": 20
            }
          ]
        }]))
        
        await handlequeryres(qid, socket, "", "all", stime)
        primalglobal(socket)
      }
      
      async function query(socket, primal){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let stime = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []

        filters.push(...(types[type] || []).map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: 20
          }
        }))
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        
        await handlequeryres(qid, socket, q, type, stime)
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []

        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = function(){
          qs("#conncount").innerText = conncount() + (primalsocket && 1)
          log("connected", this.url, conncount())
        }
        
        return socket
      }
      
      addEventListener("load", async () => {
        primalsocket = connect("wss://cache3.primal.net/cache17")
        primalsocket.addEventListener("open", function(){
          primalglobal(this)
        })
        
        primalsocket.addEventListener("close", function(){
          primalsocket = null
        })
        
        for(let url of relays()){
          const socket = connect(url)
          sockets.push(socket)
          
          socket.addEventListener("open", function(){
            checkupdate(this)
            query(this)
          })
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmUNMXPaqWFFtXe6VV8iDv85QdLhCXhUfq3EGxPQiu84bm

files.html, fix blank page with missing localStorage key

sha256

4bddf8f1eb4bf13718491c2bd9ad4f1e7b36ce9152e3410f758e468f5f9c389f images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const obj = JSON.parse(localStorage.getItem("relays"))
        const r = obj && obj.filter(r => r.match(/wss:\/\/.+/)) || []
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmNckVL17hAUjCGhhh3Mh5hgJ6HCR8bUfMAYqKBKyrEbak

sha256

91fa4738a962ef52d316e659beac4df665d5dac45da325da4e41d88b577e7977 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const r = JSON.parse(localStorage.getItem("relays")).filter(r => r.match(/wss:\/\/.+/))
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmNckVL17hAUjCGhhh3Mh5hgJ6HCR8bUfMAYqKBKyrEbak

files.html, video fixes

sha256

91fa4738a962ef52d316e659beac4df665d5dac45da325da4e41d88b577e7977 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
        margin-top: 0 !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      let hoveredvideo = null
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const r = JSON.parse(localStorage.getItem("relays")).filter(r => r.match(/wss:\/\/.+/))
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updateallvideoplay(){
        qsa("#pics video").forEach(async v => updatevideoplay(v))
      }
      
      async function updatevideoplay(v){
        const aplay = autoplay()
        
        if(aplay != "on"){
          v.muted = false
          v.pause()
        }else{
          v.muted = true
        }
        
        const n = v.parentElement
        
        if((n.offsetTop + n.offsetHeight) < scrollY 
        || n.offsetTop > (scrollY + visualViewport.height)){
          if(v.readyState == 4){
            v.pause()
          }else{
            v.preload = "none"
          }
        }else if(v.readyState != 4){
          v.preload = "auto"
        }
        else if(aplay == "on" || (aplay == "hover" && v == hoveredvideo)){
          v.play()
        }
      }
      
      addEventListener("scroll", () => updateallvideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          hoveredvideo = e.target.querySelector("video")
          hoveredvideo.play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        hoveredvideo = false
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updateallvideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          let v = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
                this.style.marginTop = -((1 / aspect * 100 - 100) / 2) + "%"
              }
              
              updatevideoplay(this)
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
          
          if(v){
            updatevideoplay(v)
          }
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay(v)
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmTFx31ocv6EZHHffSSJNc5fiQjKWuZUt3LoPKGy3V7747

files.html, update default relays to better ones (NIP-50 search)

sha256

2d6fe68072cab6793c963a303359a1778aacbd2a25e85dff576ab46b8b7954e6 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      p {
        margin-top: 0;
      }
      
      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <p>Please use relays with NIP-50 (search) functionality</p>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        const r = JSON.parse(localStorage.getItem("relays")).filter(r => r.match(/wss:\/\/.+/))
        
        return r.length > 0 && r || [
          "wss://relay.nostr.band",
          "wss://nb.relay.center", 
          "wss://nostr.compile-error.net"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmUztyyiKUJz5brYoNqYvcwi5df94YDjP7Smw5ZZHFutmm

files.html, improve connection handling

sha256

4a2bddbcb9fafcfe110647975c7d49dc8ea6cea3de555ebcaf40c93c848c4783 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        return JSON.parse(localStorage.getItem("relays")) || [
          "wss://nos.lol",
          "wss://relay.nostr.band",
          "wss://relay.mostr.pub"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmUztyyiKUJz5brYoNqYvcwi5df94YDjP7Smw5ZZHFutmm

files.html, improve connection handling

sha256

4a2bddbcb9fafcfe110647975c7d49dc8ea6cea3de555ebcaf40c93c848c4783 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        return JSON.parse(localStorage.getItem("relays")) || [
          "wss://nos.lol",
          "wss://relay.nostr.band",
          "wss://relay.mostr.pub"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", async e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        //await sleep(500).promise
        //connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(url){
        log("connecting", url)
        let socket = new WebSocket(url)
        sockets.push(socket)

        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        
        socket.onclose = async function(){
          sockets.splice(sockets.indexOf(socket), 1)
          log("disconnected", this.url, conncount())
          await sleep(1000).promise
          connect(this.url)
        }
        
        socket.onopen = async function(e){
          log("connected", this.url, conncount())
          checkupdate(this)
          query(socket)
        }
      }
      
      addEventListener("load", async () => {
        for(let url of relays()){
          connect(url)
        }
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmaSSrURfv7ToPqWjLRLudern5dG8f26oMRsgQMQTp8PW8

files.html, add: logs view, +1 default relay

sha256

7ae750a83b010b3dad7f7b1451f85bf656514dfa386006558be2c3efa11ea7e5 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays,
      #logs {
        display: block;
        width: 18em;
        height: 6em;
        background: #111;
        color: #fff;
      }
      
      #logs {
        height: 8em;
        overflow: scroll;
        font-size: .8em;
      }
      
      #relays, 
      input.submit {
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a,
      input.submit {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 18em;
        margin-left: -18em;
        background: #222;
        position: fixed;
        z-index: 3; 
        overflow: scroll;
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em 1em 1em;
        color: #fff;
      }
      
      #menu-content .items a,
      input.submit {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input class="submit" type="submit" value="save"/>
        </form>
        <div class="label">logs:</div>
        <div id="logs"></div>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        return JSON.parse(localStorage.getItem("relays")) || [
          "wss://nos.lol",
          "wss://relay.nostr.band",
          "wss://relay.mostr.pub"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        log("query", socket.url)
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        log("query res", res.length, socket.url)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        log("query meta", socket.url)
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }
  
        log("save res", results.length, socket.url)
        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        
        if(!updatenote){
          return
        }
        
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function log(msg){
        let scroll = qs("#logs").scrollTop == qs("#logs").scrollTopMax
        const div = document.createElement("div")
        div.innerText = Array.from(arguments).join(" ")
        qs("#logs").append(div)
        
        if(scroll){
          qs("#logs").scrollTo(0, qs("#logs").scrollTopMax)
        }
      }
      
      function connect(){
        for(let relay of relays()){
          if(relay.readyState == 1){
            continue
          }
          
          log("connecting", relay)
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            log("connected", this.url, conncount())
            checkupdate(this)
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmdGBuVwK51EMG84C5nJMbDfnUkoNEVqqiY1zy39aXcqzG

files.html, optimize queries

sha256

0ada4ce6f27a123e024261abc13c9cde0e76524d56379203c1b1fba2b4d1c5fe images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays {
        height: 6em;
        background: #111;
        color: #fff;
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input type="submit" value="save"/>
        </form>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 5000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        return JSON.parse(localStorage.getItem("relays")) || [
          "wss://nos.lol",
          "wss://relay.nostr.band"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#query-type").value = "all"
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = []
        
        if(type != "all"){
          filters.push(...types[type].map((t) => {
            return {
              kinds: [1], 
              search: "." + t,
              limit: l
            }
          }))
        }
        
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        
        if(filters.length == 0){
          filters.push({kinds: [1]})
        }
        
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })

        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
        reactedeventids.push(...(await querylikes(socket, eventids)))

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(socket){
        const updatesres = await querypubkey(socket, updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function connect(){
        for(let relay of relays()){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate(this)
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmP7TCL9TBmnXMGyPDaiCZbTWy8kAgmPfXF4e65oB2auaP

files.html, add: relays setting

sha256

bc55fd2a95fb6419b558d6768324fde614afdf17104588c33f76297a475bad03 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #relays {
        height: 6em;
        background: #111;
        color: #fff;
        border-radius: .5em;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">relays:</div>
        <form action="#" id="relays-form">
          <textarea id="relays"></textarea>
          <input type="submit" value="save"/>
        </form>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form id="query-form" action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      qs("#relays").value = relays().join("\n")

      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 10000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function relays(){
        return JSON.parse(localStorage.getItem("relays")) || [
          "wss://nos.lol",
          "wss://relay.nostr.band"
        ]
      }
      
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("#query-form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      qs("#relays-form").addEventListener("submit", e => {
        e.preventDefault()
        let r = JSON.stringify(qs("#relays").value.split("\n"))
        localStorage.setItem("relays", r)
        disconnect()
        connect()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = types[type].map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: l
          }
        })
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })
        
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        for(let s of sockets){
          metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
          reactedeventids.push(...(await querylikes(s, eventids)))
        }

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(){
        const updatesres = await querypubkey(sockets[0], updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function disconnect(){
        for(let socket of sockets){
          socket.close()
        }
        
        sockets.splice(0)
      }
      
      function connect(){
        for(let relay of relays()){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate()
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


this is a backup of #nostrbot dev files created with bk.sh

echo "N3q8ryccAASe2DVXUksAAAAAAAAkAAAAAAAAAACio4zhCbBFu10AMw9AgqjcSBwR5DAYUm3eOHHtLRimGMRDaYrh8zdjntDa2ndEtJEoygClBbo/LktXxWbO9y2wmisp5lWLGBagjZ0t2PMZrPV9Z3+ED+RakQOhVIwzXhuea1/mo9LslhbgJQ/793bvodS7lpzMVROGCM4nohXlqKnL2KP5GRfnkec8rvda7i877wJBVKPiA1m+5bfCZpKDfyZ62Qg7Q51V9FfH1OhQCN2AJkR69BuMQXlYXRBfLrxcVBAISEw2ipDZJA1Rm/HfpO1PETo6CPBytyimdBzWAw0jLIbEn4RzlLHTJeYZ8sYco2CFRK0Lg9wWsmCiy1zW1JwWBu/X85B1E5mMBmctg8aFhxyk2HUOJeJc0M/2RTxqB/bKDif5uPmpwn/7zmDiXPdS8xI66ugBlUdzdmcYNlmRfB/nufAvS9YGcnP9eOEZDh8XrwXkv5rmZXq93LmA0Dth3+xllyFGSmx3tm8uuFjlF3mjvfBrqyEXQ18pixc+RUtj8AuBDrZnM/6O3Q052SCEeNtS/ue1tx8PfYXGOFQmn2/aPlgWsTzOBYCKtvAtSXrvlNnymJZpNnJO7dujv9JvzUtRsvt8gEDiCbDxPEJss8G+vL34lPa62MIPIEWfF9DVMCc1HrS+L10A1ktIbnVlThQr8Tt1evakWqQzlmuFC1C97Hayv4KxbeRLQh18Um6qJzOCRJFe1tvOX7Kmgv+3HGpjhU3EmupsjkVSUsX5EUQ5eVLwYS1SpBCKeL6kVZLjuD0VDw3TUn+q8o6FPym7GVrf0GJK4sHzivild8T/z+zqc19fzizokbFmA6y6ZI9fKtl9ILHGsMpUC/mkmmKIMZC4pDJlRtIzCNt+RgAqQ0CVZ4ACNJuDdkONe1dEsQjkT5kVrkH1O+TS0sOrRDRv+oXH/HlRSlq5p+DNp13zL7/moYkC6EmUDRlg/QKTRWuTHGj5xg5xmmEOVg/QvYJHWL6j4zqhFiqyRB7suVt2H3uhs/doYsYP+ki5D/jj/uBXToAoGVWsniVC924lTvlZKUJ0/apS6BYTc9i44onh+XWfUX/dG8IbR7SToJcV12iWyJI8lYcDR/6oAePbwsXzzMD1s8aHPkOR+gpSCOZ1kaMt2eflJIrKlr8JipTMuNIISz9HHDmGvpSONGiXnUQEC1/tEubkf1kJQw5NqbttLNgnQXqS4ww+rI8BFUH3z6ay+iLKnW1AL0mO6/G33ny133kvRxv5zep+YyO/jLVK4ikZ+U4i/A+XZnDpm0xVOVCcrRJ8wcgy/Iu7eYn7MK3ySiUm39egUadXlKas09e75E7mdA5eTKWQuBnh7pSYPG3zXEVg/unEvRIRz8d83xnAyA4WpOJnbylNtYqQRBDeEq+Fa1IJVmRBP9UVX40Ypx/2OF6ERK7IoIQD+pUTsH/k39cjnZ45qk04Q61r2fuU+11/KcXDGx2YkRJjelO3AJl4GzsvvXcRyjuEnCxAuZYOEpF7xGpfBV3FAm5awa851twsK2IvyA8JlFt2Tp9C2Pr3NLFc1bvHpxX5uFmnVJG/E/eZ38Q71TlejFIEfXDC3JLkJN6/MOdpaSW3OAmBPtq13jJwX6pc5ZXvmKNS8e6b19EhLQbh58H1E3h5NdSGntkW+ppt5BZWCRjDcyOdEteEXepannOdY6Twvj4LmYExLNSZe1HXNISQuFrrcMdq5lrldVZv5hwJL1kQ2/AMLgpsQtvuhYfyyviYfRq7aTvQC0B9DJO1pfQ7C1l1YGQ62+e7R9ojCEhhXBJZWuGNA+YfGfDSCKYzex1rMbNl8mmAlWti7u6oAxGigucNgM8RsKN55piuN7QHyhWRiEL/kKISfivdSKgqLd2QhsQzx5zXIcPdDFPrX9VVDbc/7N9l9X7sgumxokEE+ZlvTPpmJ9goInx5LSNJTWMkjEUc5lwyapkdchkbh7zJEiddfzFIQpJxtZxiv0XlmzcUbdBHHr6Q/Zkezw4Jieq2NLez80qSfN3QNWkrZirgTCMXOPB7xS3IIiarP6fqh9A3dkpRAXguucHPIXtwv4CDDtGIG+MgxuJg5hQF74P5G+G8nE3DYYsg1EF3SIBo7RZR6Z7Z9r6RzD5S1YNXhvkPpGY2ZN233HuKmG4Tzpy5g6xSCGTgObEM6kY6TAijKWsTKxkA02EA9EW0ttV8d3EsanLy1WK8aci2V/dkduhHRwgcXZaEH/vSJLr/k+yDXK2xFjXtHOlkEQtM+DDBwDdAzlVPr3CkSAYdddGpAu1pAoZipuVHlgykLEDQ8xpLgyi2gw3xXYTs7WYyn7b5zSgvgjHvG9b6MJitvWS113zikLkn9oTX8AwGYo5rVzm0Qoz2AWie3f4P8oVgJZoPTenQ+UeRlM2ZQILzwkFxF/3bm8tbckA8bS+FfO3sR83hoSC8PD380RLSWYVdR5a6fcz6mtUJUk9MHcZa/oHaDmQ3EGUPzXPtVKD6h0JXUIsCdC9wjd25VlRDgzWY97j9ZzmCaWhYln2tPWyMSCEhpWtSi5Ffv8ipb//b+sx6nNMpJcwj58+2HM3q1f5b8xzeMrfVCOMNcYIjfaTm8s+nxGOjHmw5hMMXg4kny/7GqRYaGm3QcW07AW2HsHwzPqQyMOcSJrUFRf94CZoa3Gh6cF/qw8EoSI601lEw9hQlcz2IavUYvsetHmd3IrM97ukQLiwsBVQdk8Ed123lVE7BnOecYHKMB+vkyAWdB7wneGM+pmNVnnWDGCiG3jp5svqYa43V0oEkCDzAo0Xd2WzS37oKVgop3eQABMYaRIa0mmiWz5YN46dQ/YsX6oB1oXNkHBY1O99oQJX58/PdBJY+3st5fy/+STBmcO2hanJzsewavBf5I9UrH51rzhh0KuhnPFp2B8K7Lr0r+/ZBIj9Jg2UjTgCrjkY6REND6lOdFIyVtfa0uCNm8fDV64W+8xEO1Qa9avESQR8vuNNQ8FDSkkkktX0ZqaNf8ax/LO/JCSeyuGewHu+rB62LBF8ZydkIJeHWpJHUQr13t0hICUzrua2seLV0bHWLvBTO43wJqCBD6JnPsCdkWfiZwRjf1q3G1XiZG0i9Xebb8GeHdY86hOfn+syy8E91fHs1+Kz6Ydo4zOQaLayt/eazwEi3q9wNfS9QZVkkXIi9IGhrJZiGYSvCFbfGS8+34Sremzhw7SfTdidpZRCLk+ZPPCNVmp2H1KU1nzv/QDwQ8HS4MNyWUUcXJL1t+zfIVkhgYlg4taFmWYw8fzo9glxHBeLjzCevoIfovU14hKgLD54Ty1U3PoYHf8u526f8Lz+S8GmDvfYLCm7AjwL6xKCUrf8poeDpitFj+NeolPZOTSZ7xhQmKYJwMJaW616TyAgS2htTr+Yqhr0Pow1ktUnCGTajpDg5+7XapP1xNm3OAfApobiXo+Wr+tJLtI7tVKmVWQe1M7HBIlWLYlMtJ7ZrTWQmuMOpnEfd2vNZU0HSwCQO9/27FQGa8HZ+SDEZ311opd1mW9TylqUGTfGg41Jus9e0H2G6uL7WKHUXM52vzs+yi3VLo4yNFFWQr745qZaRdX9AaAnx/VFZ17UnoxaB8oQNyIQQ0d6QVFPJrINBPm8QoWVK1aM8SLMGCglnlddjckimPF9FLs2kIybOrgg8vC2ty7mNBGkIIbznSTgdyVkPZhkB8MKFpKfQxfNn64kLoIkN1VlFKdKlnzB2NL9CiIH9+J69+KhHgNQAu5Yi8fiRZmZCQMLd2mGbmud3SLh35z1YBERynd35FkpZHBcD4Hn23nH6YTK/xB2vAwI1pn3QFvRKP5fdida6XiFtMwxfn6GwgYyv4Zen7pNzbcm3alAIVZR1uJ0awhq0J6SnUejbXlOyXA18py+TbYQbqKFqcsXj4DTI3istbWoBly6Q9Mm6CeK13XaIODg5N8pqiHaQNAoIFN8Mj5p3W+wxcrvE5aTYDNnO/aVqbs1GJbBSFJ312xfufxjDw/+jFdSt8udei3nsM/5J2+owqJQGUnHAH4fsaxXUUauJhpQBhdbjgkkOKmEMOiV2MGXs1fMZP+C/0CeJ70eXFRPL/pmuFknjkWG18EC7MGwO/B9RvGJk08FqhQEx3eMmC/EgRujF6QLl/tN2+yrXTfLjsw1a4LLPL/UZGhPaFPcUcNDrecHqEvosErdn3sTMbwilwpJSPOKikjwD19ws7PrjazzEq0lsOAsGm2sqv1P2iVzSgFjY5CToeMfrw6wPxpr/EU+KumK7tgGyEK/UJttRGd4UystMomSg6L+VdILKU50dgekcewvR9//wPK+ysHnRrCmtIFzkkRKZpf5ml+RX/wa04t5E3k7MFVgk4uxUgnwPqkYwHO9WN7/Nd5u9ohvHBprTj9Gw+iKhl9X5Zzk34qVLH0aBxbu1B4yyTfC824GkDDN35PgQsp9pp9+2XpKXyu+hiU+kQpwCwRgtyafA8yvnGvNUP9LTQf+TgOD6vSHzUFPYWgxTkJVMIeKRlfl+4xs8FrBD9WVQHUxcdt4BXgg/7Mx2X7DJZecrZ2ltUgeL7mtkOjUze6XL7620HtUMcogu8sCIqlwgSu++7I/QaPiQm2bXrMSimE+FkDAZMywURJXyrpuSRpyxdef66gz+uRg0KyZvKIYzKu9RGb6IEVc1GlLDGf542KfokxzIRJZ3U+kEpPgAO/J64IzZ7UefSWK3g/4R0LvXwWLM02FefAVtXs9K6rfZHCbAjnbCJTudpbOOAAjm4DY6CtdpS30oB8/+ywJ9saLsHzLWbfgz+8KJbs9sDCoKxkN81lWEBkaTPgPiJbSd4dhpd1KjekkGOxxDmk2gqqrE+GEzCBWNZi+tqNNDcaEbHNPDY3RjU4LlAVpbO6HjVeN/N5HKAUKj32CPcZ1f2KdX+Fm/AorMooR++Fd0N8qKBm7ymexzqm8QOeyFR/nXpNTyUFULXT941qUW4A9gpnN68Ez/O55v2x7+sUqb46BnzZ8Y11RQhii9ZY4Bk6Ydh41msRjQEwulxI/nq2eKq/io0Rw5aF3GVeOlP3HPcgyEw+OuIJtwJ6fRQQ4oJO4CrcTdiCuKSzgPy0DrhDNm+Z+Kw4NliUV3nwZ23T8nhivXdJQcmnDeO+OnvTBYiu5qPy+8pSsfHCPhowneI0UfVNFjugn+eVs/pbeRsgWxTBsiEyMX1hzO+WEReFSw0x09TmW17EzXQmFf1KGXmCBskXJVC/T8JvBcpvEv0M3Nb7MUUMDP9gxdoZvaAUVWAF8/wedb3LVVVCto+pasrWpCR2+rblG/rMCSz5H9beqYN5FGld0/SRI53oVtMci4DEPuhjzxHSrPF6P8YkZlr3fGQ93ixxtY+MZ1ld6LJPmEGCucGOXixvxHixOThXRNm0o7Z7ns8ROHfHMCBCEx1oyBQ3vkwedtfGogfv4IGLIJp3IvACb6qGFTjlrd+AzBoT3aRI7HhmUTdbj4fcfVmPaGBAyRRJbfvSlMnp8e/TIfT+Dvnnrh3naiLRxr76SVV+yAOlFKXsIANoY/o55aYcTERrbmiUeE++l2apB8Ir/UiH4Tx9rTKeuVLUhYuU25ZT36FKVO0znq9MVSuvHsjGPCE1iCIx3HaDzS4tLQ5MeppDLQvkVvbJOQRAKG9bYgHv0Bfyh9zCzKKZYIYCEVojiQQIPT+3esDnNWnSWhfwNXQFluNd0VdLejGP6bOKyXqINP1GVbvvIWYL99SE2Cy5jP/w9o+qRarnPFHZTd5yVlP56s/GyhBdIDF4nWE3F/9pwhEnv8H5ND3Q9PfWzTWHr1JGBj3DTbLQYQzt8dIJxv3bznwa3L8hf6g1Rc22XndGOCMPflxVtWRsYue9Ddar/A4ZqHA99MFiu0GYuJ3puEFg8p5Hfmd0uAezJJpzxZwfSzcpFb5zE7HdNfJf+YhTeiZIZboMV8lSkTuU35P4zWRp7UKn2gnefRqgZMJWqS24Ycy/M3t2LRYeTrs2QO95FiA7vTDhorwHFgrGJ2RIAWerrkS8TelwW+Dny2o8N35aejjwIl4PR0/L6GLL1rrHKhu4tm+VPPoA8UvAdyszR7R+NGDUlENPa9razPgu99VEnMBp3U4keptC11mwIZunq9WhVfwBPl3mpfmO5kTcQkvHg9M5MuxgIaoGpvBER3c/tOtyVYmk2frjGlHEFJ0yt08Ggq+Af5L8rQex21ppRkiOkTNLr0QKgyR1bTMbYJhFqG3iQ5sGryrDCTTklNwdTMX5/56J09jSTNDsxgyE4C8+2WQ6Kc97XXa0S4ssKJgGdwV7RX4WiAAc6n/ym/xhyW+YABsXIEIySP6UB9CiGu9lR6iTpYB3Uwt8dxyxKO9jQzQ+PihUV69FjRIBJVr0MwLozc8rTGKbxL4dBm2M8WuAX7KJ1rD/1kb+KYOkon8YWvWPGvcYuAkt95M93teydN08bwVhlSjTk/GjCy42Z50h2YLKtbZG5VUrCch51cZdgKOP5bWTk7mrvN21vWOnyMAij0sJncDHov1YrwCkMS2k7y19jRxRWlrEQHX0u+eJ4QjsS/arNxqbVgCVBg8/uMTbc8YVU0VmNalsbmoLQ0UQjsOtE4Od8agn7UOUX+O52Z7DCctmiiqhDqttFsRJwrUMQx9OTBuSIDz6cTCvtnGxtZ3uzzMAdBhSmz2KfaBiHFk/U7d5bsJdjbujRLuCceDDweiLZyy1Q7KHrHOQlE3ceh9UbiGjniNS/MyP/i7bqlVoMYQ2X6WuTSMNImhQ534BnjGf0BuwG5NTD/sE8ba3Uh1TYdDljqNxQFA2U5qkvI27nCf71W1pjh/6K4qPxHTJD7bgu8EyPaY1KQZ/uerUPpoRtlkgD6IFbZ0fm/GhiiQFPkvBbvlRNzMDG+T/T1+29I6DP60fO1AURvhFSnpsGWJXEAUtuXbJtal/WANPg0WA2eIeOge+FCpP1MZaCD/9u6ST+o9Ra8UkNeEXSww8qTmeKuBth7ghlxxWCs7qNjDOs8B6kCLCzPc9N4yMfOBt8v/90cmTuC0BOce1h/XadASkMaifO1devDVibO2FAyTa6C+exKcUdJdFnOy7Y0jOlpTtUU38woOzALVFp5zWVehQhUpfag3bLX/KcuSnwr1rmqTyq2eYUSV8LeHXtqpsTif3yJH8gfjx0TJDzNznamI/1Y+BA8QW6dL9BFUipk76kukf93XDswHIJNKjaUtAW6NskRjtQhrdZGVHzi6enTxnOz9QOdeoFfNj8W2m/7aJaPJuxXW/Jupxl9aqm00Y1qVevlOZFkZWLQKLCDqQAqX42otuqRSttqj71NtyUrwPe0LBWa8jds7JWbHy1mtq0SET2D5Z8wbp+8RC1PrHFnq6TzDo9sphD83eQvw0cKFuhzeqTzt8Xr5LmSBg+VO1Ls8c9717hE9vjmr9qfKi3xCEe2flB1ehZlTATseIHuttAUOOcrH/ng1MG3c+4HIvgVYJbozmifcHPGQi/xbA+jToICiPsQqx8IZ5+tz5Lfn4b9CGBYhOfFxPH5dZFhJ21XEboL8MtPdGL7c83V0jESEY+2/q4Pv2ZzIAzI4+Fj9vsOPiJdUurgXuV1X2r5iKYl2btFk1kXJVSVp98TS0kj7ar/klCmgU/Hp+qpmlMUkzAJmD78TKQ3Q6LvB0ylq93RqbM3b0EQR6ZcCRhxGF+WQqpTiEdtgC+cRJhXH0sK1nSN4DI+haZwlJajZ1qbWb29QUCQnziAUziCYHIxaGtmfO44NI5GqSrPgItWHLWijyuOXPZYM4DaJ69jPleWF2DTlMDl6RRTGR4EDu+D5epdoDXLmSV6XCdFplLVAgLnQj7PNKTUiA2xiGRk2AfZtJMFP0PmWFOzcu9S6G9kGq+Fof2SzOu/HJ43NFZzL/USzf6FaIzUjCRx3mCXWb62NwNZPISFlt/criVAprv7etNSeeAJufwiMjr0olaQC868uBqzG3TJS6ibt+TEDOqU0sPqsHUTHia8O4nlCVA/QxImiLzJVdBffLkmIqLKA1TubD/w6iASpX/pTLxnQ5xNYgeUO/y+vMO3oGw+WKxVFDsTr1UufwUcEWGJjlwx/AYPeTcceoNga/5aKVZIM/c2D/wCtAHkqF2igIlDb9dYjuBWfiZcCxF8U6WnheCrgS867e6jXisNwpMWWtkISzaOJS4XgA5t5/d33QmLWKngIKrLefRV3rlz5QtO5cnrFGiryQNGtSxp2kyzTvqW29suAaXfq1OYAcobS+l7Fei0ipQU6H+9utlaBGlmCQNmKMfZJIHVe4irZ7FnDUnIb0M4ModO4Z7SZE/BZR72ThXLu0qVJD/aMX7yAyC10YyNS3mVLqJsXDftstFzDLBk3e1wxyJZvSbrrlgGb5sshgTOR4YvDeBxySRSywTGmzzzp3owngPJrar/O3OB5uHlgnhKy28LVWFNcjPAD+BxmotiWIrnpGr+RR6nCM2fxmzYWZgPXE0SY6GrhFTrIXNqy0wrg/KUjcSxufoVkvMmlG8pmILhA/ps8bJR5sRVb4oNOkNtzgCWBqbZjnrtdSczA6ggFQrBmxCSFLsxYVd2MWLGwurv69e+I4gPg3HRTmC2dSzMWvC8A8rDXujIXuFqF1KvRyO+de5emOJ5ytryH16ELPdZgSeNn1Hg6GHJIKdCIvYs7h17RPZHgGD1zhNlBs36+QV09x5KwT3/Dg/CmBpycfZKsQF+Yx6tZUAcVOJi7UJdJh8uCiJRbfKEWvXSKJbuR4+DwbDVEEhS5vsYHqUL+3tT2kgYY5RWBMfW5Vr2m+pT26iYkSzDzZOa64Nf5WaBZZm/184PTJnTb5syFVFJfp1C0/A0m1k9kQ9sTskTqO7esNqJU+hHoNkCa8RLeQcU12gh3XqF1PF32vDaJ3mmMSnIPbbvLkZTIWiZYNMl7BNdsyoT25kn1hkmRV13PfG5uwK1zbEqKsIyuq7RA7FvCvFDogbjUWO/xXAt5KNqywrXca743gmtUN9dqDVIY2mACSxXFiTT1KKXGkvRCu6EcnEQyeIteQrZ71ErsBp4D35kvb6hzWfUjwUV7JwDoTzAOV01ZwerjwdIlGYmtMkDNgwDDbpbQri69HJ8GEECp4xWHobTEMWhvX/A8OpOKvcmjpj2eXfytnj3CQqddb/tEFVWfMeNQQq1QszrTpF/GlbbyqWvG9oVf32yGn9ahbSkvoB906noOdiV09kf6gwbQY1hSqA6JTwFedJU75M1ovk8HtrTysZprgkSRE6v5ZnQgcZpLxC+FhhwrqRS4/NlSLoJM8vyjdxgNgFSna9fnvrf3zElRRCs841bHVWFCoVoeT8KehS2zWU6aSQdHiqj3nvanQYBst4b91NuxSpAH1q698QtWgudl50rIcmlJd4ArENB1A2EIDBgdxxORmlYQaH3d2jEN7EWfwZDufB0yjAbAjTqAePsk+vxOKCG+yz2rgNaBnJZpQSzy5EHr2SFv98LiuWT6cINlMDmCnUxpe5eebb4fqbchYGU1QxgG7w1LmnJrcr5dlBRkLslcX2uxBhZcbvahAaPYyq2/TauCTPzhK1FLtBb7sgtHuzt7+YZqjVCeDKOfrjztkBW39B790CoPmQcG0onzvnK4hiI7RzkuuVHyk6P9LiPonvNlMm+z/mzcKoNaWfcW7ZIVV26z3Owe4nQLzR15zbvIMIi3ap2IZgPmzd8EC3z8gfvFChWvg9cCmcVwjvK+daha3jXObCokQQu5HTDIOwKXu9gHP5Z7lx0WQotpEjeC2R5YjSm9a3bEX8+c18H/sr/K/n7kOPutHbC0KkySrbVV1eyZF14vcn4Uk1Ra6oXsL7gLQs2kVVuQnpN169QSKvCqzLU4Rz8TCEGv8/hUzaX8aA6ugs99Iz6TqSPS0shLRpAwgV8zoFPWhzjOw/FjEtc/bD1XPxbFDWSy+SyQ0XXrscwquvG32k6pTYCrPMn7mxp0Mx0FFYwGR/nrA/tNsSWP1NF6ID4jHasrueuh3LLnzPSOZK8DVZ49oEzCRKdZM6981bCsf4CDKUuqGfsLnZ5R2nazMW9UUHuBlDLtiHtufplieILmrlY/lTMQzs6601EBxa4qA0KWGWT02iEgZoucXgUDnJ1ECPGNVLtXSJAXycHV4VJVSXnyKCsSR0pRmWO6X1I3RJWjSc+ApdFyV3/HyZKHswyjlp/E0g+V0qu+HafykNmsWkbjY9dVC8fi0jMxisjYib8qveIijaC8BgN97so1+qDOkNqul49O3HskowxOadVJ6kRYEZifGQ2gWRQnoILYlI/q9efDIBFV+BuI3k3XLRlirQ0+JaUYcmyFMoCWRA9/MjxY80eK8m7kbUXrF43DHDsa/vSEPUQPMZY8VEQkZ9CnA7U4ehdy/MJLZrDbEF1dt6lqGchMgwO2q1ORCoSd3hNfbNvIga19FpUJiEFs4rxhvezaV2OZmBoiWgq73l2utiJz1re7Ft8Bo8+5Uqv+rWM8cjxno8J/Zeaa63w3iv6X5e8VUib34EdQvItDtey5ak84uSylPBj+jjKIsWkVSvSD7BW2h3KilcbcHjX6Yu/ov9qxiFJEHlM6XcnrPSeqqgvK8PLoWBrJhrdzPgpieuqawj+Ri8m8FBKm+ffTg3U45nmmljxkoBdWHuhUemGVzuOoSkwu9pFjy8er4hlPNZJZcHu7fLXvQt09b192m0D+8PqyPMiDu9Y+49vDt6aDCgRG3oB61qNCxxy9KSYjRg+dr6GwqrtK+VCKTnSEfw7OCqPOrKV64I59qjK9DadB5Gn+iGblCfMx+9NhYizUz0e95gh8HyT8U0jMYvpKPZZDdKGmkIdWjJY3X3EqKHRSvNqyX49S858JHGXg9QaOLLsyVfB7wUba3HPC7WTHD7uLbFe4v+NlxPaxeiy678kQbPbvO+Sf2b7V3eB3VX0rO5ZFHuFqUWmGwjH1A++NjIRE5wcJJI46ndzVcYAKMIJ1KUc/8/N/1aTfcr4gs3kmco3trQZm67LE5GUufjPMpmW3phNJZiI3M4dPnVbiLoUFPufla+9DeOT4KWlpOQ/f81sAS2ZJjwziylpQeWU0WHZFhtjepw4CdNC9cdqc6n2RoDfbCZnBcWONkCcee3axowPmzJggGeWRYWEdtieU2gX2IXFvj1i7jXb+yr63eGrljm4vTykzWKXzA/LtgGmZEKf53Gg/FRy1fDAyAgYpPvbWVJKsJ83IWpva1LBgTqIJo82wEpiiu1ofqXCgzRsPw7QUPAVV+3rnps/QAoOJivD1J1RKu1Ahpg+c9u37J2cDJIDrADaBwQ7t2253MKlZrIiqDU7Uyf0P6sNmZ6llxTcZySUyqQtN3nPMUsCJtMZuhQpKZd0MTBcDowHvnZmS7kUISLIuuFLa0yXS4Y6HFwQE9qZFhC98+MS6tWFJG6UrZPNzGKvtEyp37iFeR5Y0KKQSGaGHnOYhss87hPUpy802EwHxMVR6ez8m9NFZIfqEQ2F9I9VLzNv1+sG/k0M0oWHd9hm7AvvbGHqsrBT/KiV4BJYM3nFxh7lVux01GRaZxiJZWGF5KejxZEIfYoXhXCpsTRz5MK2ZFsPBmBDlJ7nPuBWaxzhmZMzGKrpeC7SHJvfZqzYKxuqnGeEFu/2vsFf8A3ANS1i+9cmxkbupVKB8OJVdVt6/njtfUeqLBLbNktLenXY4He0mzIlPjTEgF0tptBKV81sw7fHiOrskZST+jdHbnyTLRz6edVAWlR1iCJaDMKoLMPBQ5uCHCtPvSR35/cRt4/CAuSywbl/nSUWKtsrr5khLUNAFqZaHmir0H5GLd6wJRwFOuqVGdXPRXXyScNqE02PvGqiJx6LXAz/wLcjC/Bq9l5sMCnHV4H2JviaIxk1S2PIHYrimaRBcuGSSovLiVs8iDuwhXdMh9b0M8m1iH9+Jo2tODlzrxpfNPgx6pCs4xGV9QIKu9yWHJXdi1i96f7gEoqOmXLfpCPCxxX16tx7nKZuP8hYt38jwXOnEZXyMC6A/PThr4kdHWBtN/sJxseeSWei0xuf+3GF4W21JayHPKQwxtWFBcmZRAhZfHmZ1gs93rWdKlKghjO8mjxriu+RVtiDRNNrUHsQtOAUHkgOg4ISi4KC4a1HLaJsMAxZWPAG1Zp3xXJHFNJrUTtIBWiscCVw5eUYpn7VmInVfTyYGEAtf0N+CF63YlupgN42YjigDW4oGwNPcTUnNTTMUFWLcMznkJ+VWkPZ85zfoif8yEtHpZO2oIxDg1G7hzkA9JIiJoo8f4WcTalnbs0imtALPxmvhGJNuMnj0FZyKMZ0GNDG0u+8X2eYqSC6VGfcfRAOieufI/TGJm3jOGsvCDayVBH35FKv4CK0TM8Od5UQZVsCK1aGsaPLffaZAgl6MZ77BlRItZcZgLBKtk5pZrU9SvD3PVzEqLvhPfe9xnDkhk58U2VnQaGsdSlz3aiSjR5myU3LAR7MBThIvLow8kvuWultRgfWCgiXOBxG6yYUdysKpVD8qmbnm93Tbw6qHM4m6AvIghxTCnJBscgkYUY2RIHoO8YBIxl0wDRky72Gt2AxQ+2tPk4sWx3cs9uIcFZq5L+GhhM/aRCFmsNE4XmiT8+LzOdUPcJP5QQLSTbH9+JC8I3q1lHjV2JrmrA3rGhZ0vkdfbh1oKMK23VS11H6mRaY1pvAQdir68tAdbvQHF9sRHRfCy7PJzkYU0thGhPUOft+68JZDWdlHwtIlOBNnH7w9P0FbMgUKI/GxkScIQv3La+T2Ma2p9/2AoXGa3MDtJ6JHFdaFdFwyRYaVN9mAPHbBJ0fIUEmRYEcPkBP3usM8BgYTxcLeYxKW8vUMZEoDQugoqMzEIfdtqkXZMO5nvM7X7u9f3HjrgfSCW6oKiy0DoRGl3R1lz1ETkPGFpzzLTcLnczYVlGBngu5HQIf055XzllXyonq1Bz5VR6ZyobrCiQORWwqFrtSj5Tv7Q3QQU10WbA7jAQiEHcvI44pMXcy5T3/pQ9UluYL/wUoDIYmClCLg09q6GxBsclsOrGevMEMZohM6dEZLoqz7mBokkevEmd9S868T+7I/BfeJplM6tF+651jQmltv7Q0k+sNzD17W2QXPIUsDIyJetGMwHFbOQ+unMOcyS2pZn4j98nJrNtYBRxWn8PXBlnt3zJC559YxMDSzPwJPaNn2fLHqRr9np8ddYxatcSn8YI/VPey5Erfdh64StEJf1FaquE9DI7FbMUDu3jHTjm1aAmyvvB0npqS5PvmDFYvZBd0V6xM0/2CED33kt+vwZqDUSBg8OdvxhR1JaUS8VLIchTl0OTbz1R0VptopMSOzjL+sLoA/DME4NWa7REkQqt7nvHrC5ERume8CoqoGTWQCsxbTCTnYB7epaIKcLqNZEcbeYpd4nVxtyXpKiek7//r5Pf1hEkv0LUR1/V1z/vRug66A83nub8pUQEN7PN0wU67FHWdgOj08yIDyQnAW/PkXVg2BdKNfcfIXtwP/2+/LfqnNOGyL5ZGyywHAiLnSuFze5L8H0G2T1PF1o8ZL2J0RfNZ2fkkth+KbcfZx2h6BiP+FblGanHqmvUCTCyw6fOu4iVc0uZA0oJhsFwNPTSn7w6wXU3xMGHs6BYQEFcTgR+8NuY8TbykD076mT48HSv7xxO/Jz7/DIp5uBMXqis01ZkgBhYjaKmxvif/XDb6uXmRG28tFLpv8HGX+FFnwEqP4y/vYlBS094QYdtlfZt/c1pOy+5gyNSJf90xouD+Nle1p3Jzh2dD9/c0h2V9v9jvpkCdsKwiVNk3OueppzhgoXsxy/odghAXw04/fm7buW5n6FyZ8ZaYX/qz3Thkq145S7sgB/2CySeMFpH6AGtSH0skG+8HzS8jPhOlmxk6y2Y8ihP1GlJ7D1pfUc6hPbeY1+GDWJS4OrV0nMtkbSdD3KSU/2dj0ifdlxXUa4N4vOuIAGxGQt8i3wg0QytsghqVZ9QmKjJg28CKnrR2MCX9ILRzxEKWS6cNiCEQyrpxqECnugDUQrKN3+MlufrwLTh4HwqX/T7w5bc7Nodo1fEUOPUrrQZ8VV5BvrJwGs6x0bBQghnVrujGAtGWOm2VDxcX1Osh6t4G4bVG8s3IKwx+ixlhZzgLko5jXCI8i8K+Xol1VXYVDDHRgYNp4Sux0yF6H1S6PkItM1Y6fEQuZrK9NKGoV05vJfKGJYk5OSU2PXUS9Q3Vb7l+G8aa1Ik5Ycp7458sctE2lSUX4XBhXAwbagJaCrztC7U3PHhnB0ugaxH6bOD3u/UAoZ7zX5Dvo9PkKmB0xnRAS+2g2cxLubbWx6dGXzO6kwH/jBuYOOzXWFprI5+e18L6Nd6md4oISEfeSqNk2puzHEggXQVyaSc8NvHr9po0U48C0/h0V0Z57n5VUFk7159QCB1rjCcRNEoNwCPzi5Z6YZBoEzEnw8EhKN0605bggpG1s+3Kfs+BfES7u82gemnoMWROQ2v3X1tztTkwcrRq7U0ACZXUhDb5fDFnm07Qug5oGd7q45b1rjagT0gIxu6tARrDBDlFsXaIxeOq9ZoZFqkOfz1NWyiwxZTXYK3UxXPvDBlXzmVu2nCGyQhp6iGLIVKNnn3L+E2uOXIPk+rPEpjKDQ8n8049l7OW1v6e6rXiMwHL5j0aKuA8H8H/H01Yt1M7p7VwjXiVlZhoBDf5Ref+2/Vz/Nb3fYwBfRG+bc/6kJGec7ZQadHB0+0BSm+I5M0Kyvplb55ciBZQIYvq2TN5nIh/MjQWNPCPJd2xnkZDmWSC1mOtJ0Kh4xgnNjukHXpntUoRGZIZ3IRTm2+Hf+m35+Bgb4PkN8WvAIy/sOBU4DmGKepmIBojD56fuH2sA5ANShRS+P9jasHOqCQXhrFHgTgSq3ER/2a00P9ibURpVdh0WqpiD34NBM/XmKWbhj/ERA5E5najQ3AADwalDkT1WNnLBitU9pXvcWsaFmG+3w2cJcQcdVEmy1wW7igN7ol6DXc78RSdZ0906bAxokmJA2O7succyJ2oSriwxee4xzta4/bWhhLMYtpsdQHF+BUTueTu1Ag8oUIsU3LroCm34GASQ3ntAWTYW/HTuwA4aqmmUn30VVAYFfwrn0T2G/AjXD+moNetIAbJqsl+4yTt2sfX5+miZsq6kzGfU2pVy3y6uP9iiB5TCIgZu9lST2xhrC+41qSAuHakOpdf5cdkdxn3khnTMq76q3jhS9eeLe1gW9g+zz89XU5PFUqYQIVV8yiyhW7uAb1cojcoimDCPH7jAghF8mop7FlcIXH4G8yAOfdavzJ+ef6Nt4he/z13EFP4YU/PNQRtMrislmJX2tgN/d8wUqe0hhzAlu75I41ZBm+PXg+ByKihDd8xdVYcMt7Lo5N5NxXifGqWyveejeKv5COQ5mePRmy6es2iWofCdko9T5i1Lsxh78y2onThYr5pKZGIvF0NZQWg2Evbv1imgckXa2MamTF55AIu2hBRDasX8N//cdN8wHgXYOzegmkRxrPj71qZO70sEjZH+kvIg/Wsrc+Ks9uPlKboUJFHvTxT1ufktMXcKV0SkE4q6ukiydAr3tOuTouMSZYYaphaDERmv/HDTNTYPdMaWDq/r/BfPIOx2glZn4GYpa6z9rP/EUnUFQeyTqbYVk/2NSbS2lQZpIjbuO4RuyNM7FbYbWXdjfApK7JcVEnDMLXxVgnCkKOXbEKtFV77DCacafVtecHv+XxRo8LgX6vL/JQvVWei4t/E+858Vk9zAm3kvw8sBpJk7TGGLCvzhKRtKRuyk3wQcaV02cc8bV5VD92Mv3nuEp4vtEbeKkcj8RkYgp8Uzg5+hHVyk/boXBl4T/HjFQA4o0BFEQIa0ESU3P1wTNsfBi5wVGDqztXo+8bTnwZ1nkTA9KyR4jrZx/cpeT2FDv3D7GJskiHe/jHsGj4nHESvo0d/6/OcJUONzdp1ICUZ8J4k6/4eUXJ9bs5YFwxRFtfkghXXHW9/S8LIrGsi4c05QCw/4fikJp53I4agZ3zC1/vOno1A5Zk53zMJY9oo2gz5aX6jLqYj+Ch0vVnk4sqTflWA8ukkhfx3Yt/L+kIyey1WBnblwIRJBvydd1bNO3w0XGDCfuoJUEe57TBK3MH53+rS8F2QJMP8HRO795VkeY7D18Ci2c2uxJc0bk9mF4VLolRLVVG7sZU92twfpBhzlzjxEQRZKP7z19iBWvwdqeH/aOeFSxOCsVwPkmDFDxs+PAx3Rqj2+vIJITApzzJ77wGtKqtxv3UDJ08XrN9eGbaRUpNLi2Z4cjdW01MwZl++D6N+QeJ7s4n5ZT1jNHygamAZ6ZS0tjymaFFYsOk6hjFmRJb51hZcHgtDI1iPBMg+MDXQzacVZdgTp2YVyxR943CGKQ3owwKq1MvpXWTrHYLeoIgLph+vSYU3jEOrLlSjVL5vXYBtAk5wYOnrHPdzPoz77yol/SXKdIqEvj422/RP3LIbdWsAoU6H3SfzsVf4Gf5+KsewNZLukD4kViC/BnjkZORlsAWpBCXfu1W8f4tTJtX8XZTDux2KIuCnJLndIetTpfYrdaG18aousIAVtdSHAFoxSy/4q3BXR4jr03nV+XyxJK/EBHePUmq0JDWiLE3zStZuj947wLBvhKlRykgV+7grz/Go3+B7iV9fx3hIXosu7fhkAVrtZzKnbiz7GSe1NVCuM3XnjHf+uOpYNRy24iPgU4atl4Fz1KUNlGK/7lZ8Vq1OazWQ6u/VYPJr+4URQLZ4aQFq51nW3GLZjqn7CaY6M4vVcB0FBMwYqPFEpNzE+BzR+SPy1aODaxMnRIrTs38rd7YuZbtVFUfu0xVI3UIZwg1+RaC5UQoOPgQKS1LL/Gs8867QFlJrjcv559+qHtKsf3okIs7ztgb17g4vambZjBwaBmGP3SPlCxV/ZSUMXhtK+Nxkn/yzpmMB/+tNOq9BHAsYjaiXtFFDC+xX2nTjYUHLZzrO3wHuDmldBtv4qh7G4bBv/j0ten0bCWufsKckeEgaxbGxKuWwXJ+rHTLFSfTDl03fnIC9ckkjb37zw4Jdfzg1OHI4ycvG+1A5m1+uDv/4B6MYlxlHiSmxtsU7ToDFnw0iggRK3y+Sar3ENWlbBD4OinXpyNoiKcXvHfPfCUvcToIfmHVbVg1u9gZoXyw70378X3CNXi5u//zyE643uZEbcdoNb62Z7Y38Z5escG0rYzUvMZaqCYwQ/4tebFcvYCJZGCmQ5hZRSdadPPmGbqAu21kBnt11lzKf6iDSCHzMIDJx9uf1x+eeGI6XOeTx9MmqafcylDG7i9eaKkJLHIR9BY+GCh4uJW8zrTRSz8KIbKNW52yhD0XT2mU79JdvXAwWGTtAtDMemN9gpSEW/4UGQFHsDgMyDd5NE2pS2Wvu6M6CA4abpyVkP8itTPhNlzShwh6wvXaLk9+FNMxwxe6k5x99alFUkLkWWzKzmSFqr8w3z4blrtsjGkrnkBDhcVlgE7YdvYidCbpcGYkdylWYbRRrRhw3FfZNo7fpsZ4TC9Y5+Zba3VHbBVGTF0lBQob4U8YmqNvGNCceBi0FueIuy2cbTojcrLp1WfKs+9tqy7jt51tLupTi+pZgAjwXUl7lOrlFlZR45+F06LiSxkbJpPhuByjW8aOd8zMZZz4DrcWHWGlFPWM5KF2zH007eu/a6PmPZTAaQeVN3QBNYTjhJNs6opij2gb64iv9Jj24X2vx1FmUxUDus/F/ei633Oyh5nfNyaRNaZ0RtBrEEyxSvft2kaMwYUMtKmMQprWgUBWR+p6Hs9nGWk+rDYKc8VdkDWQw2uaVFQC/ItyIUWJTI0Bw0KNfmMOH0Jg/ykfTizuRnsQjxzDdKiWaeQ7bLlusBopxm4dWcEYxaYz8Z6Q8m52MjkRXcmvvXm/aX0+nnxW7KCvHZlYaBdB8rQmz5qyzMpoD3oPso5NpUD2px+IpbwqsXFQOOu3uyz2nAAAR/yFMBH1NnKrkKLXoJqSGcM/dbwV3dDdx7pBheDDq2DsIQ46oA62WHJkoPQYLkFKH+iaifWNd+/QbSejmCdpbkzbEd9TLDDv0QuVJgLDzqmgMRw6agQmy5RtlEuDEnuZrXSQClNBSEg7fT/PDmedqQtxdDcQOX5NMBdPXM5UmPPYQaujynR32Wqr/hsKK6b1pSJGbNN4lHOHnnNo3XG681UzPDpRaouqU8O5VPeYPIXLZTmiYs4BJmDJc7+kP9YDysJ94Gd8Ph8OitPEQfVxRYy84A/NQ+ydoawyQjjgvD1/cp2hM+pZ7NDzZPPKansaxoEdbW3BXMtR3E+4GVVa1R14JbWBIMYERMpdlBoMC3tCsJ/uU+vub46yqmjagHcDMOKXC20Nwm6IeneVpXzBC24OtoMcVjVUDQ1CuHcE8u2Clahu79dEGyh2c1END0E1bDU11k5GNYQnimSwlkzQPWLCQqTN61gcak5nAFdlYVuH8yDMbp0gYYRIAeA5zHQ0aqbwzVBAlytpZz8OEgSGRG9cjUpZgPvm5yyWhoaGbtRX7rReXxOZze+ny5SCY559sMuLRAd7PKNFjdq7hMTJP8+677NnZDOXCiENH/f/M4WxrhzkcPoLfGUwwBFAk1SIaFKbEjC8oPTHMvgMduTmH9scXwCpp7NR635aYr4VlHIsclNqXFB2845JY+N78rdkoa4Kb1jcn11A1LBoJrQ6R7ciTAtB5/hicLyxMin02yqw7xOUEJNBb3Wl0uYnQJoPCwtRL7mgfrwd8yQYII6Ni7sBKwgyDDUdfZWi8mju1KElsu5gQcGM9tayGIdskw4dcsIQxq1u6ZTYJJRsQHdm7wdVREZ9Rmq9VXA2vCtUzg4HDtVlG6i057sZa3BxdMgBDXohF0UdHcoUttXrZn6PbDv30wYMnJ9sRXM0NQvO7kcCMTu5M6c6bP/yc9O67ef1yG6B/m0HWWqSrhO1CGhCJUOQgmu/xqn5C6qne9/2CtsGTEYDsqwR/x1kDcVo4KY7F5K4Z865kMdq5U2mM2o+7yxMdi0Uas2hek5k1db6ZljInzLhc22Wkg3h/hwc4VVBephBZhkE9hlhRlhu7e3asouweD3zPYCXFh7B+K2OkdyNPNbh4KxyhFJovaRaruHQkq1hSfE3g8YuaaF8zvSDhZT6IXSyOeZB3S3N2QnoJbLE/lPvWHYme3XWslvXR+l43A72og+Ag1qlzZQQXyRCntvko7oUHMgdstnIWgATEv55YSPCFy/aTQNteaWDPzc/rBdzyUV6sHhTcOwi4lnqefRWraOpd30DIPl4PV37qkPm1jLxWrenAHvq8jG5e6tJeGsvlCzYpVjMMa3VFazNiGS/QxKfTk3sQxDspQn14UgXYCPTvUYzz5/atm4BlDkrykPxiWZvNrM0VbbiOaef9TWtrSrfR3N/MaHAiZ0KQHe4oku8avbTPKfJPbJLlEHGVDocmF95dr7sJ0CIDLPIgtxZtOiUNUWDDmQft14hj8xMSNqOmRpOjPOBwNW6MmMwoExT3q8UaNnfkh2S7cifO4Iye9vkHR4PJWfiUH6jivbvX6e1kIP9wURWfPxDFaVc8ehBlLeTq+z6XVxVoghu7/5OpP9/4ug/viegom8FIvTsny1fHssvN/7lvix1ZMeDFFSBxJhjmgYvD1qesg39fWCqwhC0+OBn9Nz+CPlPyhPHZylijl23JtnWhcJkeoBR/dJyFDJ6l3kGTjE7JkXDshcQR325+9+05M1T0GCj+290N7Z7s7HlJ+LitPG9ynBKK31/Nvt8E9N0MB1bj5QsFD+l9PhMZQEbOqwXusslCHbt9wmfXLtZTzan0VIY64pIl/3ecikiqncgE4rplSymsnLiKA/4hzJDpT1MD1BcTCVdPMv7ByLZ9K2U8WTQ5XjTfHb589nxWyCu0J5pr4W0g/RIQgQpE5DgXOu9djiYh6AMvbI6fvz+kzsUHXH91ofK4PwWTckh6VvPz6FXL4ZYIwHx+zxqqxLm3GRcjfHfZED1Qr+PLBhpIsxxUgTuyy68FSEp2e++vN7NHWqWhKO/nP69QSOtgOW62YSwfJwMQJ9aPfO1/EzINxjTNA6HNE4ouy/0QWuxPPY4jvj3rEcGjiFgAbgDbY60fKPLrGr4vtYN0vQ8Y4/0DLYj4y7uHxlsb5RNiKdNj8kjaul4KuhraUkDjZZJDiNDXBSEt9NAYBZNnZnQl2HQVmjcXfKBvFjfU+a2SQXyFJwhbY2NiOtsGDTh6Gm/Vp6Zn8XhjnxZc/7OtyfNfPzmssPsY5Usb3aKhinjyYP0Xy0JwLCbZbqqtc34bguP8Xh83R4aeZHOzlHEh3H+xQF40mIq6u9RUH8yDtlIR2qQTYqD9yQHs3oVymyVaJ/Hoo1Rwk1nWgH/qZ1EOosUgNFt6MfS4Pev435pCxmwhatyAvUN7hP8ylqkshFCNngedCJfnDGfoLe3BH7nyp1ylsng/oOtFlN9CsyVs8qMBzTPvjrdwkhqgI70bDgPRIbazBZXrT5N+CPL3HVSMPL7JNYzij1ktftg5WLi4Nz6Wc6k+7g0toYebM+Zj5t/dO9LUF32wkm8ZprohSP+RXHixwxDkH2FSrdHzKHHuOOVn+6wFTcgLMVLOl68wdXZwWVRd4jQ8I4Jdm8TytUZMyMV8eqDzHObJhTC8JG1wjeGPvr5sTK6J1ggxgB6+MhRZmXvX5LVEtiV6ch+1O1stmnQHlid8D2UuPXUNXVVwXjEa2759NHW7pwyauKV1yLJjfdQdXLPvKwRaNaG4aHespFgb/yNtmPQA+TN7c6MZaVdBR5vomZrPQWUBEiXcLegteuCS3vDhH6FvYKLpk/+vd+9bBMvNmZpe9iaBvO7GP3sXbeNtKMuSQ6UB1/IZ9kjQu79CAG+AE3/N8RyJXQ6ZnpesViWNQtooNgEj25LbbLdT3pXZiRp23auqGCrO2+G6MqwMffx4KWAXs9ihdWiBpu7nSj7HvbhHT72xm66pwyHFlmF8/SrrxtQNhOZUFZFIsKpsuoGxusYLLjy1jjsbiks+U3vgNF1B01T0nD87yoDD3JoRBk9cFqsnYHL2LQWo5+zi+m+x+QraursMOwfHYeW2EXHFN7EG6QPHoPjUErpK7z0ZmLUYUybU6pXxax0AkZEfs9fS99b8MXZRNUUu0/JGDuxKp+lPZRSxNvyvwuA1cxLtbKc4FFw7n4EI3/qLaOeUou3tAMKHhAGIzFyH+/+uzkuLLcDWM42FdCFNQg/r5441BTTg17KZKy37cfZcQic9TQqAFlcqRDJ2/F0DLFFPFbOs8umRAuoTkOoHyQodDLlnSzI65XZesF/C0g/G76HDf3dvVcTAUHrp1C5LQEOWXHWYCJ5wRExlThYYRin1vSImGxQoMNw1Ga5y0/CmQ9OXnmwomcRJcXIeW0iIh9qkowM1I0lPSULiQuAZJ57sH4sJxNsl5lItqPuTBtIblhZ4kAiGwoqEFOWkgNrOfKIx9w27KEnkneo+DuB0JMeJZn8hXNlDQHj9bphjlBKi+1rnhC4Yh/7VUX3PQuSpPdS6hMzHa587oUvkyFkk0votGwHubnd6ZcfBoE9L6wyUy6ElPpOtO4MeIXQe3Zowd1u8TeOsSISsgOyXXOBuVYA5GgL9LiaU0IABxf6ix4pnzzC5oAKJuYgrF3GCAak915Y0w+KI52XlcYtQpSOUNxbX2i/RAUQdC9m91+s2O7qu7fAmIEnPVWL9dPWZbdiEnhJl+XjYMhUUBD0vhmpQ0f0mu+7qxiYEiIzSdVzi6VSjWBIkFVNhEp9NluDLP3XwKfYFnRG/DXEZZzFLVF6inmbxuOy0VLovJF0C4g31eAnxhkPtMmOvQNQ4k59TiZAvR42dkKGZkLhNjv2Lbgo6YlsuoKSRHg3fD56/azXpGMXD2I1gLDK5/k9l41DG08XP2PeptlzAxmtj9Zs666AOv0cf7Xxss3x9K3kIdwPmj+suysF8QRHHKAVjvhki6f8NKkyRgprgDtbH6OW7c/q/OuioBZpPm9KDrsD33nsWKHmqp71gBUBWig4F9RubiHz7Rf1dv8hcs3G+s1ZXhoZBSSw1a6P7OMv6zgILOwl+N4qydL7KACmdwFo7sxyMZN9xr9xSiigUaGKwyXsa5NUddBr8hIP5icvLbtbJDFi1Mt1uFNjeP4dz4o8umuYXtJLYkAgLUlJSIX51SILSMkyiywMc3h43wahli1xGzyIVpTjFGdMTJjtlHc8mnLidLIElbRiPB0PFwOX5yd7lUMnixT2cms3mJrwvdiYImLa57mL88YuGkXUHLuYg8uI9II2ayPXfcp7MhN0NPE4DCROPBNcHoCNVrjjlbxv+WbWteVlz3wcWA8mccSE3HiQuQpNJ4m4zGgfXUuXUyAqhroqWwnSA+69SbhLYUxAwe3QgNL1YUqzkI8y6TinWLHR1Ms91MXpoeXrRA001i63Sq6f3c74SB/ojlFO7koY3nLhgpKnIbDc5shZksLCG+aQz8LsK+BfaDCH1OCZWqQ3VC3XQ5p/DMxypY4R19cpn8hb43U1pCdPbJ1ixvIOKfRUQ+uzK6gFG7uR2F1aW/7q57It1utDIpOgwyqwOK1LD3VVH1y+vHJGbTLXd4TtiwoJ0mM6G51PiCyA47pG4UPYi5KgNwKY/wY9WvtvtjG/L0wpYuVOJbcqQFnP6SCGxd+M2KmicA5iq79+x7t4vsiD0ItES7XZFvp6oPm2Jw7mFxvDTXrrbxTsau5WMl+J1nLA3Fid7mFi+/UG5crcqDmnFTnh4HgeeRuG0jsPvj/T75ZOtbkn0UJYwt1qpYNMIWlNHwcqjY49JnNRK8uWc3TcdDiH06CBMIGmo5kdwVwUqKbviSFRIIjZrQ6j4CL3QceoqWnqQaSKk92US6/79EWaz9dljaAaBJljEfA7jCblkjD/Q9Alo4nT07LIQ2yhCdxHtPEx6Lpa94OZoJYbVby/QWHwfAv53C2y2BuSS7Ymwny8v3n+eVjJ4Otk0NCU8U0dPaKCe57dz/zwiC5exQS1c4dKlogPfRyMVJn/lmFUD0XVcrKBV4byA/x+tSB8SwIJJ6uySOH7wYDPt902C0gH6mFwwoXlWiyHHYglf8BNDxtuILevmP/EaaUD7WNxfli8ifPyG9dgFLpIcw5MDDPchymLzkKGG1p3bqqB2b7aJcWSfu4F/feX6y0meh+yU4Olhvn01huE4Jwq8RJ0t9Lonp/Bwdt+BmmPq5LDOKnMI8kLbkvhhSz4wQY+7CRq7lO9fcKtrvdMTO3XNbM9NBsnI8D91qoYinsOzxvmBXERP7Jo+OZMn4TitXdNXzlnuqyS3yJKMCFQKwwkFXeNstj25flr8wEVB9xFC13aOilWrZ/PJzELdtzW47BgUArZ9OKWFI6BTuiPOwJb6GWdWTFKTo4+rr8ACptdFKNZ1/ojZNNEWOfJ2IbAUsgNYqBAc8SVXrRt+BCH2ogXQCmphosu09gwE/NJ3JMZatpZZgs7zPdQPenl1JyIf3f6/GylA38MwEZ9PrrW49qKUrtdAuouh/9V9xqT8SWrf+3DyQ52NbxqvN7C36AkXqxHXfkcIrLf3jr1pA2znBptegu9maI2JP60LHiDsP8iTd7LTFwA+FT2MMr/jJ1z3M67LB5pL2l3cMsWZBXLI5GWqNjC2OjXuE5rB/SMCiFcM4RVrxJpyHIbtaS9KNNjJ9RztCL4/unHFBFepIslTOTbWyt+GirKAkE+lq+xuvvUfYyO8iMqRk/x0F7To1u7ib4kjboxP3UuGLJGJs0ZOd/W5mNFRE7VtEqUwNgRqgaypHCNbu8u6cfAX1pUojszknne390VjkgHteAAAAIEzB64P1/FETwGL5qNzmioglsXcUVPxKT+34+mCAm1Tl//Z6KCJY2KvI7CsU/BlkzpRYDlr3BBqB6DIZ71cKvHhZHgLMicITdLqhCu3aFHk0CLt3++bkl4rx9ugsHzSdWDjqXnRqqtpVupDb1Hgov7wM/5lPBbKZEVFXLt5EKbCxjJhMn+J88Jq3Zy78kXgOtWWAbRJWDhtt9JCd1O3tWL4OAdv4Rw7QxM/RnYHIRV8s4MM7KH7mNLcsM1CZXgXVZLlpT1xyzZaCVXJcC+AL10iTuyq7a+WKthM0NzkgZE4uc019g/ctWgSb/Rf+NmmF0bm8JrxioM3dJpOEPqEZz0u/rMDRhByj/afknNXJPo3QAN92gkBKF1h2ybKXR/MNGLO5WD7KGGV/l5vsPYpu3KByrUbdObQYva/ktluRssYkYtCmZQ+cDOv7n9xZq2OC+0wdFdq9WuCGwu29SlUN7zOVP6owB6xk22wKVKD+x/zR5XFC/v8Zxlad7oeqKlR7Ggm8lLGvybqWsrPS/aoS0hrsDVIFa5wEeCI9YHwi6w98Vrw4b30QAMG4SFH7yXfqeNMUDJO0SQYe504XYWL8xz04jrTP4pUjNW+MNcBQ8LTH/WU1rh7jjD8orj/yZxJVMVqIjUOWjOAZmoxNH0Mfubk5iZPQ6Z9MHeDPX/4Rxcb0cFVY0XX8x7gPp0lbcSGrCsSWaixSgiNFMJfh08jHf+C+bIBkBDG5jKvGt2KNh4prwpLxGtHeR8GusTLwELcK62r4HzGE4lRQZNkn3PL8Czv2Hl3omYO9dYaTh+ZpGtNXHrzLHaiK9NknQlYYlrSRlrTAduowiUm8h9PivFlMGflirQ6m6/joGAAC3bde/ebNolMMyhkb+KV2yJn0ZDbIvfiuotG7npVCm8zynxx8QhYqpUTKI3JXsTJLPwFT/PIlVdKbgHXl9uMvXYgZyMxxr0pu6nbnfRJhHa/oEJhUcWB+X+ErD75I2qgYI/8e8Q6CFY+BnFwHCGVZ2W18nKnfRQhhvbUaADmi49Ntg0G1hvNcyRom6H8DYkUTD/+2Kubf78bbxdVKh6MhQY1nECABVZPhG2DAOe7XuC8tWQ/y4eRuNfT8T21uHUFLmLavJ6NO44yBNm5VVaiEZ9o7j4rBzK0XhXiwsaWr7KGLE2mUUk3msXPC/aZLqoFP+Iyz9uUus/oLqpayC9avWwDPtYJ63RKPTwiwZ3AIEKyri/W+BrT+7I1AD9oOjOzN22Dg8u67KhsWPvk1Zpz956lnfS7hG+Fjjs5shktIsTpwyFBO4Lj/inii1KXb6v4kAYSgho4pK5KpO4J5EYnsvP2mDrvmVr5UTGN6uJyXEyE0Q6EnzZtKoj10HyBnYQf8Tr8JGQ8KRqcuJDhTVRkg9oV+OebbpIzFsemFAa7ZMvNl7/ST1sg0e2G0WlWPynTcLyIOJ/k0UeyMoaWQEJ+bU22LABP441+qRMHZQiJ/04uu5iWNrwo0TXeUowfsjhdNcUcE+tSir1dozo+Oj4M5QW5nrsnqgfmZtxkk5xRoxMTnAIz016e6OPFODajg+/D6v7WVvGnZmbrKt4JYAxrIxgzhS3Se0NzKX9blaU5i3ZKKFV+IwyBiQ8xYI5+y81keG/GJehCyHAGaHH8ofGgstDhgE7PjdcLEOqzSHT81PoVLioWbM/PmezT3kdrZEXLy0I7gTPH/V5XlB21gxXTLbFRRumW1WUkjiKyJPHlZ0iCyo9GVG5oUm1+S8W2Ob/Gxnz2hJ8bn94cXs1l/x0105VabKBU5MqTjN8ooFua2e4WUv5PmeiIB50r8BWbjtOyLl1MxpRj1qUvkYRfxHaTWsn7ZJKvNG1Mq2uqrPBWIrTgOtGwnSvUW9+8OMaKpkvjmQAAFwbAw0UBCYWPAAcLAQABIwMBAQVdABAAAAyPggoBGBMvpgAA" | base64 --decode > bk.7z && 7z x -ofiles bk.7z

shasum -a 256 bk.7z

3263e022ea4b8d94fead7b264c7ecf13b7e775631fd9deb239325776f0fe58ef

18.8 kB

https://ipfs.io/ipfs/QmVZY8fHQVKj75fjw1VzKG4A6H6bTkvfYsNFFCAFq5bHN8

files.html, fix: like filter

sha256

401420d27fe489a8ea8486b71d0aa6199cebc81819797a744dfae21a678531c1 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      .like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      const relays = [
        "wss://nos.lol",
        "wss://relay.nostr.band/"
      ]
      
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 10000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = types[type].map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: l
          }
        })
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })
        
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        for(let s of sockets){
          metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
          reactedeventids.push(...(await querylikes(s, eventids)))
        }

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'
          let root = null
          
          if(item.url.match(videoextregex)){
            root = document.createElement("div")
            root.classList.add("video-container")
            root.innerHTML = contentelements
            
            const v = document.createElement("video")
            v["data-item-id"] = id
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            //root.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            root.append(v)
          }else{
            root = document.createElement("a")
            root.href = "#"
            root.innerHTML = contentelements
              
            if(type == "pic"){
              root.innerHTML += '<img onload="gid(\'' + id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              root.innerHTML += '<span class="file">' + item.url + '</span>'
            }
          }
          
          root.id = id
          root["data-item"] = item
          root["data-event"] = item.event
          root["data-query"] = q
          root["data-text"] = text
          
          root.classList.add(type)
          root.classList.add("item")
          root.classList.add("show")
          
          if(item.reacted){
            root.classList.add("reacted")
          }
          
          qs("#pics").prepend(root)
        })

        if(active() == null){
          qs("#pics .item")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(){
        const updatesres = await querypubkey(sockets[0], updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function connect(){
        for(let relay of relays){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate()
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/Qmbi7Bj1PsEwdNCsK5SMoySu5SThNKYxnoyoM5W4Pkeo1s

files.html, fix: nostr error (blank page)

sha256

7a9b56782d64776e5488b1433a44c7a4826d55c66efd904be835440b70ddcfcd images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      a.like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      const relays = [
        "wss://nos.lol",
        "wss://relay.nostr.band/"
      ]
      
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 10000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = types[type].map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: l
          }
        })
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids, [mypubkey])
        
        return events.map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })
        
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        for(let s of sockets){
          metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
          reactedeventids.push(...(await querylikes(s, eventids)))
        }

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'

          if(item.url.match(videoextregex)){
            const div = document.createElement("div")
            div.classList.add("video-container")
            
            const v = document.createElement("video")
            div.classList.add("item")
            div.classList.add("show")
            
            div.id = id
            div["data-item"] = item
            div["data-event"] = item.event
            div["data-query"] = q
            div["data-text"] = text
            div.innerHTML = contentelements
            v["data-item-id"] = id
            
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            div.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            div.classList.add("video")
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            div.append(v)
            qs("#pics").prepend(div)
          }else{ 
            const a = document.createElement("a")
            a.id = id
            a.classList.add("item")
            a.classList.add("show")
            a.classList.add(type)
            
            if(item.reacted){
              a.classList.add("reacted")
            }
            
            a["data-item"] = item
            a["data-event"] = item.event
            a["data-query"] = q
            a["data-text"] = text
            a.href = "#"
            a.innerHTML = contentelements
              
            if(type == "pic"){
              a.innerHTML += '<img onload="gid(\'' + a.id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              a.innerHTML += '<span class="file">' + item.url + '</span>'
            }
            
            qs("#pics").prepend(a)
          }
        })

        if(active() == null){
          qs("#pics a")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(){
        const updatesres = await querypubkey(sockets[0], updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function connect(){
        for(let relay of relays){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate()
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmfSMmDP51P2VCp8j8btZcmSNfTHweaRzJokiMh7n7gA64

files.html, fix: show profile username

sha256

08ef52fc486031cc85a40ad66c4c7057bdfd3fb7cdeccac3048b11c06e6292d1 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > .item:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      a.like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      const relays = [
        "wss://nos.lol",
        "wss://relay.nostr.band/"
      ]
      
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 10000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = types[type].map((t) => {
          return {
            kinds: [1], 
            search: "." + t,
            limit: l
          }
        })
        if(q.length > 0){
          filters.push({kinds: [1], search: q})
        }
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(socket, res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        const mypubkey = await nostr.getPublicKey()
        const events = await queryevents(socket, [7], eventids)
        
        return events.filter(e => e.pubkey == mypubkey)
          .map(e => e.tags.find(t => t[0] == "e")[1])
      }

      async function queryevents(socket, kinds, eventids, authors){
        if(!window.nostr){
          return []
        }
        
        const queryid = "events_" + (++idcounter)
        const filters = {}
        
        filters.kinds = kinds
        
        if(eventids){
          filters["#e"] = eventids
        }
        
        if(authors){
          filters["authors"] = authors
        }
        
        const json = JSON.stringify([
          "REQ", 
          queryid, 
          filters
        ])
        
        socket.send(json)
        
        return await waitres(queryid) || []
      }
      
      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(socket, eventdb, q, qtype){
        let eventpubkeys = []
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          if(o.kind == 1){
            eventpubkeys.includes(o.pubkey) || eventpubkeys.push(o.pubkey)
            return true
          }
          
          return false
        })
        
        
        let results = []
        const eventids = events.map(e => e.id)
        let metaevents = []
        let reactedeventids = []
        
        for(let s of sockets){
          metaevents.push(...(await queryevents(socket, [0], null, eventpubkeys)))
          reactedeventids.push(...(await querylikes(s, eventids)))
        }

        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'

          if(item.url.match(videoextregex)){
            const div = document.createElement("div")
            div.classList.add("video-container")
            
            const v = document.createElement("video")
            div.classList.add("item")
            div.classList.add("show")
            
            div.id = id
            div["data-item"] = item
            div["data-event"] = item.event
            div["data-query"] = q
            div["data-text"] = text
            div.innerHTML = contentelements
            v["data-item-id"] = id
            
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            div.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            div.classList.add("video")
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            div.append(v)
            qs("#pics").prepend(div)
          }else{ 
            const a = document.createElement("a")
            a.id = id
            a.classList.add("item")
            a.classList.add("show")
            a.classList.add(type)
            
            if(item.reacted){
              a.classList.add("reacted")
            }
            
            a["data-item"] = item
            a["data-event"] = item.event
            a["data-query"] = q
            a["data-text"] = text
            a.href = "#"
            a.innerHTML = contentelements
              
            if(type == "pic"){
              a.innerHTML += '<img onload="gid(\'' + a.id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              a.innerHTML += '<span class="file">' + item.url + '</span>'
            }
            
            qs("#pics").prepend(a)
          }
        })

        if(active() == null){
          qs("#pics a")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(){
        const updatesres = await querypubkey(sockets[0], updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function connect(){
        for(let relay of relays){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate()
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>


https://ipfs.io/ipfs/QmcFK31zVkPx1WdpXMmxiNW77n1YiBZ7PN4o419Sn3DVD9

files.html, mobile optimizations (swipe left & right)

sha256

e5314e8ecf41a238c36fda393c6aa4be8ddf211fc334f56e16b08fbb0c2d5840 images.html

base64

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <meta charset="UTF-8"/>
    <link rel="stylesheet" href="http://svgicons.sparkk.fr/svgicons.css"/>
    <style>
      body {
        margin: 0;
        padding: 0;
        width: 100vw;
        overflow-x: hidden;
        font-family: "Noto Serif";
      }
      
      #header a, 
      form, 
      #querydisplay, 
      #header {
        background: #000;
      }
      
      #querydisplay .text, 
      #header a,
      #media-container {
        color: #fff;
      }

      #header a, 
      form, 
      #querydisplay {
        z-index: 1;
        position: relative;
      }
      
      @media only screen and (max-width: 30em) {
        #querydisplay,
        #controlinfo {
          display: none;
        }
      }

      form {
        display: inline-block;
      }
      
      #query {
        width: 6em;
      }
      
      #loading {
        display: none;
        position: absolute;
        top: 0px;
        left: 27em;
        color: #fff;
      }
      
      a {
        text-decoration: none;
      }

      video {
        width: 100%;
        height: 100%;
      }
  
      #media-container video {
        width: 100% !important;
        height: 100% !important;
      }
      
      .video-container {
        /*cursor: pointer;*/
      }
      
      .type {
        position: absolute;
        color: #fff;
        z-index: 1;
      }
      
      #pic-query {
        width: 5em;
      }
      
      #querydisplay, 
      #controlinfo,
      a {
        color: #999;
      }
      
      #querydisplay {
        padding: 0 1em;
      }

      #querydisplay .text {
        display: inline-block;
        min-width: 3em;
        padding: 0;
      }

      #header a,
      #header span {
        padding: 0.8em;
        position: relative;
        background: #000;
        color: #fff;
      }

      #header {
        padding: 0.45em 0;
        margin: 0;
        overflow: hidden;
        white-space: nowrap;
      }

      #header .items {
        background: #000;
        position: relative;
        float: left;
      }
      
      #header .items, 
      #controlinfo {
        margin-top: .3em;
      }

      .label {
        padding: .8em .8em .8em 0;
      }

      span#controlinfo, 
      #pics a > a,
      #pics .query {
        position: absolute;
      }
      
      span#controlinfo {
        right: 2em;
        white-space: nowrap;
        padding: 0 1em;
        position: absolute;
      }

      #media-container {
        position: fixed;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0);
        display: none;
        z-index: 4;
      }
      
      #pic-container {
        height: 100%;
      }
      
      #pic {
        max-width: 100vw;
        max-height: 100vh;
      }
        
      #pics {
        overflow: hidden;
      }
      
      #pics > a, 
      #pics img,
      #pics .file,
      #pics .video-container {
        width: 25vw;
      }

      #pics > a,
      #pics .file,
      #pics .video-container {
        height: 25vw;
        display: none;
        position: relative;
        float: left;
        overflow: hidden;
        border: 0.3em solid transparent;
        border-radius: 0.3em;
        margin: -0.3em;
      }

      #pics > .file {
        display: block;
      }
      
      #media-container video {
        width: 100%;
        height: 100%;
        position: absolute;
        top: 0;
        left: 0;
      }
      
      body:not(.nostr) #filter-likes, 
      .filter-likes #pics > a:not(.reacted) {
        display: none !important;
      }
      
      .query, 
      .like, 
      #pic-query,
      #pic-user {
        z-index: 2;
        display: block;
        text-align: right;
        color: #fff;
        bottom: 0;
        right: 0;
        border-radius: 3vw;
      }
      
      .query, 
      .like {
        position: absolute;
        padding: 0 1vw;
      }
      
      .query, 
      .like, 
      #pic-query {
        bottom: 0;
        font-size: min(3.5vw, 1.5em);
        background: rgba(0,0,0,0.6);
        max-width: 5em;
      }
      
      #pic-query {
        padding: .6em .4em .1em .4em;
        max-width: 5em;
      }
      
      a.like.reacted,
      .filter-likes #filter-likes {
        background: red;
      }

      #pic-query,
      #pic-user {
        font-size: 3em;
      }
      
      #pic-query,
      #pic-user,
      #media-footer .like {
        z-index: 5;
        position: fixed;
      }
      
      #pic-query,
      #pic-user {
        display: block;
      }

      #pic-user {
        bottom: 1.9em;
        font-size: 1.8em;
        padding-right: .65em;
        max-width: 8em;
        overflow-x: hidden;
        white-space: nowrap;
        text-overflow: ellipsis;
      }
      
      .like {
        display: none;
        left: 0;
        right: auto;
        padding: .2em;
        width: 1em;
        height: 1em;
        border-radius: 50%;
        text-align: center;
      }

      .nostr .like {
        display: block;  
      }

      #media-footer .like {
        font-size: 2.4em;
      }

      #media-footer {
        display: none;  
      }
  
      #pics img {
        display: block;
        position: relative;
        float: left;
      }

      #menu-content .items .active {
        background: #444;
        color: #fff;
      }
      
      #pics > .active {
        border-color: green;
        z-index: 1;
      }

      .heartsvg path {
        stroke: #fff;
      }
  
      a#menu, a#filter-likes {
        display: block;
        margin-top: -.3em;
        background: #000;
        text-align: right;
      }

      a#menu {
        float: left;
        padding: 0 .3em 0 .5em;
        height: 2.1em;
      }
      
      a#filter-likes {
        padding: .2em;
        border-radius: 50%;
        width: 1.5em;
        height: 1.5em;
        margin: .1em .5em;
        position: absolute;
        right: 0;
      }
      
      a#filter-likes svg {
        width: 1.5em;
        height: 1.5em;
      }
       
      .menusvg {
        width: 2em;
        height: 2em;
      }

      .menusvg path {
        fill: #fff;
      }

      #menu-content {
        float: left;
        height: 100vh;
        width: 15em;
        margin-left: -15em;
        background: #222;
        position: absolute;
        z-index: 3; 
      }

      #menu-content.open {
        margin-left: 0;
      }
      
      #menu-content .items {
        padding: 0 1em;
        color: #fff;
      }
      
      #menu-content .items a {
        padding: .5em .7em;
        display: inline-block;
        background: #111;
      }

      #menu-overlay {
        display: none;
        background: #000;
        height: 100%;
        width: 100%;
        z-index: 3;
        position: fixed;
        opacity: .5;
      }

      #menu-overlay.open {
        display: block;
      }
      
      svg, 
      img, 
      .video video {
        pointer-events: none;
      }

      #template {
        display: none;
      }
    </style>
  </head>
  <body>
    <div id="template">
      <svg class="heartsvg svg-icon" viewBox="0 0 20 20">
        <path d="M9.719,17.073l-6.562-6.51c-0.27-0.268-0.504-0.567-0.696-0.888C1.385,7.89,1.67,5.613,3.155,4.14c0.864-0.856,2.012-1.329,3.233-1.329c1.924,0,3.115,1.12,3.612,1.752c0.499-0.634,1.689-1.752,3.612-1.752c1.221,0,2.369,0.472,3.233,1.329c1.484,1.473,1.771,3.75,0.693,5.537c-0.19,0.32-0.425,0.618-0.695,0.887l-6.562,6.51C10.125,17.229,9.875,17.229,9.719,17.073 M6.388,3.61C5.379,3.61,4.431,4,3.717,4.707C2.495,5.92,2.259,7.794,3.145,9.265c0.158,0.265,0.351,0.51,0.574,0.731L10,16.228l6.281-6.232c0.224-0.221,0.416-0.466,0.573-0.729c0.887-1.472,0.651-3.346-0.571-4.56C15.57,4,14.621,3.61,13.612,3.61c-1.43,0-2.639,0.786-3.268,1.863c-0.154,0.264-0.536,0.264-0.69,0C9.029,4.397,7.82,3.61,6.388,3.61"></path>
      </svg>
      <div id="colscss">
      .cols${cols} #pics > a, 
      .cols${cols} #pics img,
      .cols${cols} #pics .file,
      .cols${cols} #pics .video-container {
        width: calc(100vw / ${cols});
      }

      .cols${cols} #pics > a,
      .cols${cols} #pics > .file,
      .cols${cols} #pics > .video-container {
        height: calc(100vw / ${cols});
      }
      
      .cols${cols} .query,
      .cols${cols} .like {
        font-size: min(14vw * (1 / ${cols}), 6em * (1 / ${cols}))
      }
      </div>
    </div>
    <div id="media-footer">
      <a id="pic-like" class="like" href="#"></a>
      <a id="pic-query" href="#"></a>
      <a id="pic-user" href="#"></a>
    </div>
    <center id="media-container">
      <div id="pic-container">
        <img id="pic"/>
      </div>
    </center>
    <div id="menu-overlay"></div>
    <div id="menu-content" class="test">
      <div class="items">
        <div class="label">autoplay:</div>
        <a class="setautoplay" data-autoplay="on" href="#">on</a>
        <a class="setautoplay" data-autoplay="off" href="#">off</a>
        <a class="setautoplay" data-autoplay="hover" href="#">mouseover</a>
        <div class="label">cols:</div>
        <a class="setcols" data-cols="2" href="#">2</a>
        <a class="setcols" data-cols="4" href="#">4</a>
        <a class="setcols" data-cols="6" href="#">6</a>
        <a class="setcols" data-cols="10" href="#">10</a>
        <div class="label">limit:</div>
        <a class="setlimit" data-limit="15" href="#">15</a>
        <a class="setlimit" data-limit="30" href="#">30</a>
        <a class="setlimit" data-limit="100" href="#">100</a>
        <div class="label">latest release:</div>
        <a id="release" href="#"></a>
      </div>
    </div>
    <div id="header">
      <span id="controlinfo">controls: arrows, enter, esc</span>
      <a id="filter-likes" href="#"></a>
      <div class="items">
        <a id="menu" href="#">
          <svg class="menusvg svg-icon" viewBox="0 0 20 20">
		        <path d="M10,1.445c-4.726,0-8.555,3.829-8.555,8.555c0,4.725,3.829,8.555,8.555,8.555c4.725,0,8.555-3.83,8.555-8.555C18.555,5.274,14.725,1.445,10,1.445 M10,17.654c-4.221,0-7.654-3.434-7.654-7.654c0-4.221,3.433-7.654,7.654-7.654c4.222,0,7.654,3.433,7.654,7.654C17.654,14.221,14.222,17.654,10,17.654 M14.39,10c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,9.55,14.39,9.752,14.39,10 M14.39,12.702c0,0.247-0.203,0.449-0.45,0.449H6.06c-0.248,0-0.45-0.202-0.45-0.449c0-0.248,0.203-0.451,0.45-0.451h7.879C14.187,12.251,14.39,12.454,14.39,12.702 M14.39,7.298c0,0.248-0.203,0.45-0.45,0.45H6.06c-0.248,0-0.45-0.203-0.45-0.45s0.203-0.45,0.45-0.45h7.879C14.187,6.848,14.39,7.051,14.39,7.298"></path>
	        </svg>
        </a>
        <a id="title" href="#">files.html</a>
        <form action="#">
          <input type="text" id="query" name="query" id="" />
          <select id="query-type">
            <option value="all">all</option>
            <option value="image">images</option>
            <option value="video">videos</option>
          </select>
          <input id="go" type="submit" value="go"/>
        </form>
        <span id="querydisplay">
          <span class="text"></span>
        </span>
        <div id="loading">updating</div>
      </div>
    </div>
    <div id="pics"></div>
    <script src="https://slowli.github.io/bech32-buffer/assets/js/bech32-buffer.min.js"></script>
    <script>
      const updatespubkey = "ae5c6e0b74660d2194c8254b5c2c825676576be0297379fa20523114a6a85e87"
      const queries = {
        "global": '["REQ","<I>",{"cache":["explore",{"timeframe":"latest","scope":"global","limit":<L>}]}]', 
        "author": '["REQ","<I>",{"kinds":[1],"authors":["<P>"]}]',
        "reactions": '["REQ","<I>", {"kinds":[7],"#e": <E>}]'
      }
      const relays = [
        "wss://nos.lol",
        "wss://relay.nostr.band/"
      ]
      
      const colopts = [2, 4, 6, 10]

      cols = () => parseInt(setting("cols", colopts[1]), 10)
      autoplay = () => setting("autoplay", "off")
      limit = () => parseInt(setting("limit", 15), 10)
      
      document.body.classList.add("cols" + cols())
      
      let protocol = navigator.userAgent.match(/Mobile/) ? "nostr:" : "web+nostr:"
      let sockets = []
      let reshandlers = []
      let urls = []
      let reacted = []
      let idcounter = 0
      let querywait = null
      let types = {
        image: ["png", "jpg", "jpeg", "webp"],
        video: ["mp4", "webm"]
      }
      types["all"] = [...types.image, ...types.video]
      let imgregex = /http(|s)\:\/\/[^ \n]*?\.(png|jpg|jpeg|webp)/gi
      let videoregex = /http(|s)\:\/\/[^ \n]*?\.(mp4|webm)/gi
      let videoextregex = /\.(mp4|webm)$/i
      waitms = 10000
      querystr = ""
      lastquerystr = ""
      lastquerytype = ""
      
      updatequery()
      qs("#pic-like").innerHTML = qs("#filter-likes").innerHTML = qs("#template .heartsvg").outerHTML
      setactive("setcols", "cols", cols())
      setactive("setautoplay", "autoplay", autoplay())
      setactive("setlimit", "limit", limit())

      function fromHexString(str){
        let buffer = new Uint8Array(str.length / 2)
        for (let i = 0; i < buffer.length; i++) {
          buffer[i] = parseInt(str.substr(2 * i, 2), 16)
        }
        return buffer
      }
  
      function setting(key, defaultvalue){
        const v = localStorage.getItem(key)
        return v && option("set" + key, key, v) && v || defaultvalue
      }
      
      function option(c, key, val){
        return qs("." + c + "[data-" + key + "='" + val + "']")
      }
      
      function qs(q){
        return document.querySelector(q)
      }
      
      function qsa(q){
        return Array.from(document.querySelectorAll(q))
      }
      
      function gid(i){
        return document.getElementById(i)
      }
      
      function querytype(){
        return qs("#query-type").value || "all"
      }
      
      function updatequery(){
        let parts = decodeURIComponent(location.hash.substring(1)).split("/")
        querystr = qs("#querydisplay .text").innerText = parts[0]
        qs("#query").value = parts[0]
        qs("#query-type").value = parts[1] || "all"
      }

      function active(){
        return qs("#pics .active")
      }
      
      function picoffset(offset){
        const pics = qsa("#pics > .item").filter(e => e.style.display == "block")
        return pics[pics.indexOf(active())+offset]
      }

      function lastactive(){
        return qsa("#pics .active").pop()      
      }

      function menu(){
        qsa("#menu-content, #menu-overlay").forEach(n => n.classList.toggle("open"))
      }
  
      function setactive(c, key, val){
        qsa("." + c).forEach(n => n.classList.remove("active"))
        const option = qs("." + c + "[data-" + key + "='" + val + "']")
        option?.classList.add("active")
      }
      
      async function updatevideoplay(){
        let enabledvideos = 0
        let totalvideos = 0
        let readyvideos = 0
        
        qsa("#pics video").forEach(async v => {
          totalvideos++
          
          if(v.readyState == 4){
            readyvideos++
          }
          
          const aplay = autoplay()
          if(aplay != "on"){
            v.pause()
          }
          
          const n = v.parentElement
          
          if((n.offsetTop + n.offsetHeight) < scrollY 
          || n.offsetTop > (scrollY + visualViewport.height)){
            if(v.readyState == 4){
              v.pause()
            }else{
              v.preload = "none"
            }
          }else{
            enabledvideos++
            if(v.readyState != 4){
              v.preload = "auto"
            }
            
            if((aplay == "on" || aplay == "hover") && v.readyState == 4){
              v.play()
            }
          }
        })
      }
      
      addEventListener("scroll", () => updatevideoplay())
      
      let touch = null
      
      addEventListener("touchstart", e => {
        touch = e.changedTouches[0]
      })
      
      addEventListener("touchend", e => {
        const xdiff = e.changedTouches[0].screenX - touch.screenX
        const ydiff = e.changedTouches[0].screenY - touch.screenY
        
        if(xdiff < -50){
          next()
        }
        
        if(xdiff > 50){
          prev()
        }
      })
      
      qs("form").addEventListener("submit", e => {
        e.preventDefault()
        location.hash = "#" + encodeURIComponent(qs("#query").value + "/" + querytype())
        updatequery()
        querywait.resolve()
      })

      addEventListener("mouseover", async (e) => {
        if(autoplay() == "hover" && e.target.classList.contains("video-container")){
          e.target.querySelector("video").play()
        }
      })
      
      addEventListener("mouseout", async (e) => {
        if(qs("#media-container").style.display != "block" && autoplay() == "hover" && e.target.classList.contains("video-container")){
          qsa("video").forEach(v => v.pause())
        }
      })
      
      addEventListener("click", async (e) => {
        //console.log("click", e, e.target.href)
        if(e.target.type == "submit" || (e.target.tagName == "A" && e.target.href.match(/^(|web\+)nostr\:/))){
          return
        }
    
        if(e.target.id == "title"){
          qs("#query").value = ""
          qs("#go").click()
        }

        e.preventDefault()

        if(e.target.id == "filter-likes"){
          document.body.classList.toggle("filter-likes")
          return
        }
        
        if(e.target.classList.contains("show")){
          active()?.classList.remove("active")
          gid(e.target.id).classList.add("active")
          show(e.target.id)
          return
        }
        
        if(e.target.tagName != "A" && (qs("#pic-container") == e.target || qs("#media-container video") == e.target)){
          hidemedia()
          return
        }

        if(e.target.id == "menu" || e.target.id == "menu-overlay"){
          menu()
          return
        }

        if(e.target.classList.contains("setcols")){
          for(let c of colopts){
            document.body.classList.remove("cols" + c)
          }
          
          document.body.classList.add("cols" + e.target.dataset.cols)
          localStorage.setItem("cols", e.target.dataset.cols)
          setactive("setcols", "cols", cols())
          return
        }

        if(e.target.classList.contains("setlimit")){
          localStorage.setItem("limit", e.target.dataset.limit)
          setactive("setlimit", "limit", limit())
          return
        }
  
        if(e.target.classList.contains("setautoplay")){
          localStorage.setItem("autoplay", e.target.dataset.autoplay)
          setactive("setautoplay", "autoplay", autoplay())
          updatevideoplay()
          return
        }
        
        if(e.target.classList.contains("like")){
          like(e.target)
        }
      })
      
      async function like(link){
        const event = link.parentElement["data-event"]
        const pubkey = await nostr.getPublicKey()
        const like = await nostr.signEvent({
          "kind": 7,
          "content": "+",
          "tags": [
            ["e", event.id],
            ["p", event.pubkey]
          ],
          "pubkey": pubkey,
          "created_at": parseInt(Date.now() / 1000 , 10)
        })
        
        sockets.forEach(s => s.send(JSON.stringify(["EVENT", like])))

        link.classList.add("reacted")

        const picid = link["data-item-id"] || link.dataset.picId
        
        if(picid){
          gid(picid).querySelector(".like").classList.add("reacted")
          reacted.push(picid)
        }
        
        console.log("reaction sent")
      }
    
      function prev(){
        if(picoffset(-1)){
          picoffset(-1).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }
      }
      
      function next(){
        if(picoffset(1)){
          picoffset(1).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }
      }
      
      function updateshow(){
        if(qs("#media-container").style.display == "block"){
          hidevideo()
          show(active().id)
        }
      }
      
      addEventListener("keydown", e => {
        if(["Tab", "ArrowLeft", "ArrowRight", "ArrowUp", "ArrowDown"].includes(e.key)){
          e.preventDefault()
        }

        if(e.key == "Escape"){
          menu()
          return
        }
        
        if(e.key == "Enter" && e.target.tagName == "BODY"){
          if(qs("#media-container").style.display == "block"){
            hidemedia()
            return
          }
          
          show(active().id)
          return
        }

        if(e.key == "ArrowUp" && picoffset(-cols())){
          picoffset(-cols()).classList.add("active")
          lastactive().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowDown" && picoffset(cols())){
          picoffset(cols()).classList.add("active")
          active().classList.remove("active")
          updateshow()
        }

        if(e.key == "ArrowLeft"){
          prev()
        }

        if(e.key == "ArrowRight"){
          next()
        }
      })

      async function query(socket){
        qs("#loading").style.display = "block"
        
        let t = Date.now()
        let q = querystr.trim()
        let type = querytype()
        let l = limit()
        let i = ++idcounter
        let qid = "query_" + i
        let filters = types[type].map((t) => {
          return {
            kinds: [0, 1], 
            search: "." + t,
            limit: l
          }
        })
        if(q.length > 0){
          filters.push({kinds: [0, 1], search: q})
        }
        const searchobj = ["REQ", qid, ...filters]
        const searchjson = JSON.stringify(searchobj)

        socket.send(searchjson)
        const res = await waitres(qid)
        save(res, q, type)
        qs("#loading").style.display = "none"
        let wait = waitms - (Date.now() - t)
        querywait = sleep(wait)
        await querywait.promise
        query(socket)
      }
      
      async function querylikes(socket, eventids){
        if(!window.nostr){
          return []
        }
        
        const mypubkey = await nostr.getPublicKey()
        
        const queryid = "likes_" + (++idcounter)
        socket.send(queries.reactions.replace("<I>", queryid).replace("<E>", JSON.stringify(eventids)))
        const res = await waitres(queryid)
        return res && res.filter(e => e.pubkey == mypubkey).map(e => e.tags.find(t => t[0] == "e")[1]) || []
      }

      async function querypubkey(socket, pubkey){
        const queryid = "query_" + (++idcounter)
        socket.send(queries.author.replace("<I>", queryid).replace("<P>", pubkey))

        return (await waitres(queryid))
      }

      function saveresults(event, regex, metaevents, reactedeventids){
        let results = []
        
        for(let url of event.content.match(regex) || []){
          if(!urls.includes(url)){
            const meta = metaevents.find(o => o.pubkey == event.pubkey)
            
            urls.push(url)
            results.push({
              reacted: reactedeventids.includes(event.id), 
              url: url, 
              event: event, 
              bech32id: bech32.encode("note", fromHexString(event.id)),
              meta: meta
            })
          }
        }
        
        return results
      }
      
      async function save(eventdb, q, qtype){
        const metaevents = eventdb.filter(o => o.kind == 0)
        const events = eventdb.filter(o => {
          if(q.length > 0){
            for(let word of q.split(" ")){
              if(o.content.toLowerCase().indexOf(word.toLowerCase()) == -1){
                return false
              }
            }
          }
          
          return (o.kind == 1)
        })
        
        let results = []
        
        const eventids = events.map(e => e.id)
        let reactedeventids = []
        for(let s of sockets){
          reactedeventids.push(...(await querylikes(s, eventids)))
        }
        
        for(let event of events){
          if(qtype != "video"){
            results.push(...saveresults(event, imgregex, metaevents, reactedeventids))
          }

          if(qtype != "image"){
            results.push(...saveresults(event, videoregex, metaevents, reactedeventids))
          }
        }

        updatecontent(results, q)
      }
      
      function updatecontent(results, q){
        results.reverse().forEach(item => {
          const type = item.url.match(videoextregex) && "video" || "pic"
          const text = (q || "view note").substring(-30)
          const id = "item_" + (++idcounter)
          const contentelements = '<div class="type">' + type + '</div><a class="like' + (item.reacted ? ' reacted' : '') + '" data-item-id="' + id + 
            '" href="#">' + qs("#template .heartsvg").outerHTML + '</a><a class="query" href="' + protocol + 
            item.bech32id + '">' + text + '</a>'

          if(item.url.match(videoextregex)){
            const div = document.createElement("div")
            div.classList.add("video-container")
            
            const v = document.createElement("video")
            div.classList.add("item")
            div.classList.add("show")
            
            div.id = id
            div["data-item"] = item
            div["data-event"] = item.event
            div["data-query"] = q
            div["data-text"] = text
            div.innerHTML = contentelements
            v["data-item-id"] = id
            
            v.preload = "none"
            v.loop = true
            v.muted = true
            v.playsinline = ""
            div.style.display = "block"
            
            v.oncanplay = async function() {
              const aspect = this.videoWidth / this.videoHeight
              if(aspect > 1){
                this.style.width = (aspect * 100) + "%"
              }else{
                this.style.height = (1 / aspect * 100) + "%"
              }
              
              updatevideoplay()
              this.parentElement.style.display = "block"
            }
      
            div.classList.add("video")
            const ext = item.url.match(videoextregex)[1]
            v.innerHTML = '<source src="' + item.url + '" type="video/' + ext + '"></source>'
            
            div.append(v)
            qs("#pics").prepend(div)
          }else{ 
            const a = document.createElement("a")
            a.id = id
            a.classList.add("item")
            a.classList.add("show")
            a.classList.add(type)
            
            if(item.reacted){
              a.classList.add("reacted")
            }
            
            a["data-item"] = item
            a["data-event"] = item.event
            a["data-query"] = q
            a["data-text"] = text
            a.href = "#"
            a.innerHTML = contentelements
              
            if(type == "pic"){
              a.innerHTML += '<img onload="gid(\'' + a.id + '\').style.display=\'block\'" src="' + item.url + '"/>'
            }else{
              a.innerHTML += '<span class="file">' + item.url + '</span>'
            }
            
            qs("#pics").prepend(a)
          }
        })

        if(active() == null){
          qs("#pics a")?.classList.add("active")
        }
        
        updatevideoplay()
      }

      function hidemedia(){
        document.body.style.overflow = "scroll"
        qsa("#media-container, #pic, #media-footer").forEach(n => n.style.display = "none")
        gid("pic-like").classList.remove("reacted")
        hidevideo()
      }

      function hidevideo(){
        const v = qs("#media-container video")
        
        if(v){
          v.muted = true
          gid(v["data-item-id"]).appendChild(v)
          updatevideoplay()
        }
      }
      
      function waitres(key){
        let events = []
        
        let reshandler = function(resolve, reject){
          this.handle = function(res){
            this.key = key
            let o = JSON.parse(res.data)
            if(o[1] == key){
              if(o[0] == "EOSE"){
                resolve(events)
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              events.push(o[2])
            }
          }
        }
          
        return new Promise((resolve, reject) => {
          reshandlers.push(new reshandler(resolve, reject))
        })
      }
    
      async function checkupdate(){
        const updatesres = await querypubkey(sockets[0], updatespubkey)
        const updatenote = updatesres.find(n => n.content.match(/^https\:\/\//))
        const age = Math.round((Date.now() / 1000 - updatenote.created_at) / 3600 / 24)
        const noteid = bech32.encode("note", fromHexString(updatenote.id))
        qs("#release").href = protocol + noteid
        qs("#release").innerText = (age == 0 && "today" || age + " day(s) ago")
      }

      function sleep(ms){
        return new function(){
          let timeout = null
          let r = null
          
          return {
            promise: new Promise(function(resolve, reject){
              timeout = setTimeout(() => resolve(), ms)
              r = resolve
            }),
            resolve: function(){
              clearTimeout(timeout)
              r()
            }
          }
        }
      }
      
      function show(id){
        document.body.style.overflow = "hidden"
        const item = gid(id)["data-item"]
        const media = gid("media-container")
        
        if(item.url.match(videoextregex)){
          const v = gid(id).querySelector("video")
          qs("#media-container").appendChild(v)
          v.muted = false
          v.play()
        }else{
          const pic = gid("pic")
          pic.src = item.url
          pic.onload = () => (pic.style.display = "block")
        }
        
        if(item.meta){
          const metacontent = JSON.parse(item.meta.content)
          const name = metacontent.display_name || metacontent.name
          qs("#pic-user").href = protocol + bech32.encode("npub", fromHexString(item.meta.pubkey))
          qs("#pic-user").innerText = name
        }

        gid("pic-like")["data-item-id"] = id
        gid("media-footer")["data-event"] = item.event

        if(item.reacted || reacted.includes(id)){
          gid("pic-like").classList.add("reacted")
        }
        
        const a = qs("#pic-query")
        a.href = protocol + item.bech32id
        a.innerText = gid(id)["data-text"]
          
        media.style.display = gid("media-footer").style.display = "block"
      }
      
      function conncount(){
        return sockets.filter(r => r.readyState == 1).length
      }
      
      function connect(){
        for(let relay of relays){
          if(relay.readyState == 1){
            continue
          }
          
          let socket = new WebSocket(relay)
          console.log("connecting to", socket.url)
          sockets.push(socket)
          socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
          socket.onclose = async function(){
            console.log("disconnected", this.url, conncount())
          }
          socket.onopen = async function(e){
            console.log("connected", this.url, conncount())
            checkupdate()
            query(socket)
          }
        }
      }
      
      addEventListener("load", async () => {
        connect()
        
        await sleep(100).promise
        
        if(window.nostr){
          document.body.classList.add("nostr")
        }
      })
      
      for(let c of colopts){
        document.write('<style>' + colscss.innerText.replace(/\$\{cols\}/g, c) + '</style>')
      }
    </script>
  </body>
</html>
