Styled and animated zaps

This commit is contained in:
ekzyis 2024-12-27 18:02:34 +01:00
parent 4eaf0e1003
commit 56236d5409
9 changed files with 96 additions and 20 deletions

1
.gitignore vendored
View File

@ -2,3 +2,4 @@
zaply zaply
*_templ.go *_templ.go
__livereload __livereload
public/css/tailwind.css

View File

@ -5,4 +5,5 @@ dev:
build: build:
templ generate templ generate
tailwindcss -i ./public/css/input.css -o ./public/css/tailwind.css
go build -o zaply main.go go build -o zaply main.go

30
components/zap.templ Normal file
View File

@ -0,0 +1,30 @@
package components
import (
"fmt"
"github.com/ekzyis/zaply/lightning"
)
templ Zap(inv *lightning.Invoice) {
<div class="bg-[#212529] w-fit zap-animate-in border border-[#212529] m-3 rounded-lg" id={ inv.PaymentHash }>
<div class="flex flex-row gap-3 items-center">
<svg width="32" height="32" viewBox="0 0 200 307" fill="#fada5e" xmlns="http://www.w3.org/2000/svg" class="ps-3">
<path d="M56 0L107.606 131H90.2129H89L1.52588e-05 131L177 307L106.979 165H121H160H200L56 0Z"/>
</svg>
<div class="flex flex-col pe-3 py-1">
<div class="text-lg text-[#f0f0f0]">{ inv.Description }</div>
<div class="text-sm text-slate-300">{ fmt.Sprintf("%.8s / %s", inv.PaymentHash, humanize(inv.Msats)) }</div>
</div>
</div>
</div>
}
func humanize(msats int64) string {
sats := msats / 1000
if sats == 1 {
return fmt.Sprintf("%d sat", sats)
} else {
return fmt.Sprintf("%d sats", sats)
}
}

View File

@ -1,7 +1,7 @@
#!/usr/bin/env bash #!/usr/bin/env bash
PID=$(pidof zaply) PID=$(pidof zaply)
DIRS="env/ lightning/ lnurl/ pages/ server/" DIRS="components/ env/ lightning/ lnurl/ pages/ server/"
set -e set -e

View File

@ -6,27 +6,25 @@ templ Overlay() {
<meta charset="UTF-8"> <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>zaply</title> <title>zaply</title>
<link href="/css/tailwind.css" rel="stylesheet">
<link href="/css/zap.css" rel="stylesheet">
<script src={ GetBaseUrl(ctx) + "/js/htmx.min.js" } integrity="sha384-HGfztofotfshcF7+8n44JQL2oJmowVChPTg48S+jvZoztPfvwD79OC/LTtG6dMp+" crossorigin="anonymous"></script> <script src={ GetBaseUrl(ctx) + "/js/htmx.min.js" } integrity="sha384-HGfztofotfshcF7+8n44JQL2oJmowVChPTg48S+jvZoztPfvwD79OC/LTtG6dMp+" crossorigin="anonymous"></script>
<script src={ GetBaseUrl(ctx) + "/js/htmx-sse.js" } crossorigin="anonymous"></script> <script src={ GetBaseUrl(ctx) + "/js/htmx-sse.js" } crossorigin="anonymous"></script>
if GetEnv(ctx) == "development" { if GetEnv(ctx) == "development" {
<script src={ GetBaseUrl(ctx) + "/js/livereload.js" }></script> <script src={ GetBaseUrl(ctx) + "/js/livereload.js" }></script>
} }
</head>
<body>
<div hx-ext="sse" sse-connect="/overlay/sse" sse-swap="zap" hx-swap="beforeend" class="fixed bottom-0 right-0" />
<script> <script>
const sse = new EventSource("/overlay/sse"); document.body.addEventListener('htmx:sseMessage', function (e) {
sse.onmessage = (event) => { setTimeout(() => {
// console.log("event", event) const div = document.getElementById(e.detail.lastEventId)
}; div.classList.add('zap-animate-out')
sse.addEventListener("zap", event => { div.addEventListener('animationend', div.remove)
let inv }, 60_000)
try {
inv = JSON.parse(event.data)
} catch(err) {
console.error("error parsing zap event", err)
return
}
console.log("zap received:", inv.paymentHash, inv.msats, inv.description)
}) })
</script> </script>
</head> </body>
</html> </html>
} }

3
public/css/input.css Normal file
View File

@ -0,0 +1,3 @@
@tailwind base;
@tailwind components;
@tailwind utilities;

26
public/css/zap.css Normal file
View File

@ -0,0 +1,26 @@
.zap-animate-in {
animation: slide-in 0.5s ease-out;
}
.zap-animate-out {
animation: fade-out 0.5s ease-out;
}
@keyframes slide-in {
0% {
transform: translateY(100%);
}
100% {
transform: translateY(0%)
}
}
@keyframes fade-out {
0% {
opacity: 1;
}
100% {
opacity: 0;
}
}

View File

@ -2,17 +2,19 @@ package server
import ( import (
"bytes" "bytes"
"encoding/json"
"fmt" "fmt"
"io" "io"
"log" "log"
"time" "time"
"github.com/a-h/templ"
"github.com/ekzyis/zaply/components"
"github.com/ekzyis/zaply/lightning" "github.com/ekzyis/zaply/lightning"
"github.com/labstack/echo/v4" "github.com/labstack/echo/v4"
) )
type Event struct { type Event struct {
Id []byte
Event []byte Event []byte
Data []byte Data []byte
} }
@ -28,6 +30,10 @@ func (ev *Event) MarshalTo(w io.Writer) error {
} }
} }
if _, err := fmt.Fprintf(w, "id: %s\n", ev.Id); err != nil {
return err
}
if _, err := fmt.Fprint(w, "\n"); err != nil { if _, err := fmt.Fprint(w, "\n"); err != nil {
return err return err
} }
@ -61,14 +67,17 @@ func sseHandler(invSrc chan *lightning.Invoice) echo.HandlerFunc {
return err return err
} }
case inv := <-invSrc: case inv := <-invSrc:
data, err := json.Marshal(inv) buf := templ.GetBuffer()
if err != nil { defer templ.ReleaseBuffer(buf)
if err := components.Zap(inv).Render(c.Request().Context(), buf); err != nil {
return err return err
} }
event := Event{ event := Event{
Id: []byte(inv.PaymentHash),
Event: []byte("zap"), Event: []byte("zap"),
Data: data, Data: buf.Bytes(),
} }
log.Printf("sending zap event: %s", inv.PaymentHash) log.Printf("sending zap event: %s", inv.PaymentHash)

8
tailwind.config.js Normal file
View File

@ -0,0 +1,8 @@
/** @type {import('tailwindcss').Config} */
module.exports = {
content: ["./components/**/*.templ", "./pages/**/*.templ"],
theme: {
extend: {},
},
plugins: [],
}