diff options
| author | Paul Buetow <paul@buetow.org> | 2026-04-10 10:23:20 +0300 |
|---|---|---|
| committer | Paul Buetow <paul@buetow.org> | 2026-04-10 10:23:20 +0300 |
| commit | f40fee44e8f256328ca1419863b5441123a1014e (patch) | |
| tree | b3a5eabc0b8ac0801240544392edaadf5a6d8ac4 /internal/generator | |
| parent | bc45b7af3bc93ccd3e4359d29e93417d0af407e1 (diff) | |
Release v0.1.0v0.1.0
Splash: skip via ?splash=0 on pagination to index; frosted panel and vignette
for readable copy; brighter hint/tag colors. Pagination links only at bottom
of each page. Tests updated for prev href to index.
Made-with: Cursor
Diffstat (limited to 'internal/generator')
| -rw-r--r-- | internal/generator/doc.go | 3 | ||||
| -rw-r--r-- | internal/generator/generator.go | 11 | ||||
| -rw-r--r-- | internal/generator/generator_test.go | 4 | ||||
| -rw-r--r-- | internal/generator/shared.go | 139 | ||||
| -rw-r--r-- | internal/generator/theme_aurora.go | 59 | ||||
| -rw-r--r-- | internal/generator/theme_brutalist.go | 49 | ||||
| -rw-r--r-- | internal/generator/theme_glass.go | 68 | ||||
| -rw-r--r-- | internal/generator/theme_matrix.go | 61 | ||||
| -rw-r--r-- | internal/generator/theme_minimal.go | 63 | ||||
| -rw-r--r-- | internal/generator/theme_neon.go | 67 | ||||
| -rw-r--r-- | internal/generator/theme_ocean.go | 57 | ||||
| -rw-r--r-- | internal/generator/theme_paper.go | 60 | ||||
| -rw-r--r-- | internal/generator/theme_retro.go | 54 | ||||
| -rw-r--r-- | internal/generator/theme_synthwave.go | 72 | ||||
| -rw-r--r-- | internal/generator/theme_terminal.go | 48 |
15 files changed, 782 insertions, 33 deletions
diff --git a/internal/generator/doc.go b/internal/generator/doc.go index 55db6cb..b22ede6 100644 --- a/internal/generator/doc.go +++ b/internal/generator/doc.go @@ -9,7 +9,8 @@ // - themes.go — Theme registry (name → template string) and getTheme / // ListThemes for the CLI. // - theme_*.go — One file per visual theme: full-page HTML that invokes -// {{template "navhints" .}}, {{template "navmodal" .}}, {{template "navscript" .}}. +// {{template "splashGate"}}, {{template "navhints" .}}, {{template "navmodal" .}}, +// {{template "navscript" .}}. // - shared.go — navDefs: shared {{define}} blocks merged at parse time with // the chosen theme so a single html/template parse sees every name. // - templates.go — Short pointer: where templates and registry live. diff --git a/internal/generator/generator.go b/internal/generator/generator.go index 9fed673..e880ba3 100644 --- a/internal/generator/generator.go +++ b/internal/generator/generator.go @@ -114,6 +114,11 @@ func pageFilename(index int) string { return fmt.Sprintf("page%d.html", index+1) } +// indexPageNavURL is the href for pagination links to the first page. splash=0 +// is read by splashGate so the splash does not run (referrer is unreliable for +// keyboard / programmatic navigation from page2.html → index.html). +const indexPageNavURL = "index.html?splash=0" + // writePage renders one HTML page and writes it to cfg.OutputDir. func writePage(tmpl *template.Template, posts []*post.Post, pageIndex, totalPages int, cfg *config.Config) error { data := buildPageData(posts, pageIndex, totalPages) @@ -148,7 +153,11 @@ func buildPageData(posts []*post.Post, pageIndex, totalPages int) pageData { // "Prev" means newer — page index decreases. if pageIndex > 0 { - prevPage = pageFilename(pageIndex - 1) + if pageIndex == 1 { + prevPage = indexPageNavURL + } else { + prevPage = pageFilename(pageIndex - 1) + } } // "Next" means older — page index increases. diff --git a/internal/generator/generator_test.go b/internal/generator/generator_test.go index ba0964e..cfab15e 100644 --- a/internal/generator/generator_test.go +++ b/internal/generator/generator_test.go @@ -130,9 +130,9 @@ func TestBuildPageData_navLinks(t *testing.T) { name: "middle", pageIndex: 1, totalPages: 3, - wantPrev: "index.html", + wantPrev: "index.html?splash=0", wantNext: "page3.html", - wantPrevJSON: `"index.html"`, + wantPrevJSON: `"index.html?splash=0"`, wantNextJSON: `"page3.html"`, wantPostsCount: 1, }, diff --git a/internal/generator/shared.go b/internal/generator/shared.go index a34c5a5..583851c 100644 --- a/internal/generator/shared.go +++ b/internal/generator/shared.go @@ -1,15 +1,48 @@ package generator // navDefs is appended to every theme template when parsing. -// It defines three named sub-templates shared across all themes: +// It defines named sub-templates shared across all themes: +// - "splashGate" — synchronous script: first child of <body>; sets html.sno-splash-skip when +// splash should not run (?splash=0, not index.html, or Referer from same-site index/pageN). // - "navhints" — keyboard shortcut hint bar HTML // - "navmodal" — full-screen expanded-post modal HTML + image-sizing CSS // - "navscript" — keyboard navigation JavaScript with distinct sounds per action // -// Each theme calls {{template "navhints" .}}, {{template "navmodal" .}}, and -// {{template "navscript" .}} at the appropriate points in its HTML. +// Each theme calls {{template "splashGate"}}, {{template "navhints" .}}, {{template "navmodal" .}}, +// and {{template "navscript" .}} at the appropriate points in its HTML. // All theme-specific CSS lives in each theme file so themes stay self-contained. const navDefs = ` +{{define "splashGate"}} +<script> +(function(){ + try { + var sp = new URLSearchParams(location.search); + if (sp.get('splash') === '0') { + document.documentElement.classList.add('sno-splash-skip'); + return; + } + } catch (_) {} + var parts = location.pathname.split('/').filter(function(s) { return s.length; }); + var seg = (parts.length ? parts[parts.length - 1] : '').toLowerCase(); + var onIndex = (!seg || seg === 'index.html'); + var ref = document.referrer; + function refIsSameSiteBlogPage(url) { + if (!url) return false; + try { + var ru = new URL(url), cu = new URL(location.href); + if (ru.origin !== cu.origin) return false; + var rp = ru.pathname.split('/').filter(function(s) { return s.length; }); + var rs = (rp.length ? rp[rp.length - 1] : '').toLowerCase(); + if (rs === 'index.html' || rs === '') return true; + if (/^page\d+\.html$/.test(rs)) return true; + return false; + } catch (_) { return false; } + } + if (!onIndex || refIsSameSiteBlogPage(ref)) document.documentElement.classList.add('sno-splash-skip'); +})(); +</script> +{{end}} + {{define "navhints"}} <div class="nav-hints" aria-label="keyboard shortcuts"> <span><kbd>j</kbd><kbd>k</kbd> or <kbd>↑</kbd><kbd>↓</kbd> select post</span> @@ -29,6 +62,37 @@ const navDefs = ` .post-modal { background:rgba(0,0,0,0.55) !important; backdrop-filter:blur(6px) !important; } /* Content area max-width across all themes */ .overlay { max-width:1200px; margin-left:auto; margin-right:auto; } +/* Pagination: newer + older side by side at the bottom of the feed */ +.page-nav-dual { display:flex; justify-content:center; align-items:center; flex-wrap:wrap; + gap:clamp(16px,4vw,48px); } +/* Host note under the site subtitle (all themes) */ +.logo-host { font-size:0.65rem; opacity:0.55; margin-top:4px; letter-spacing:0.3px; line-height:1.3; } +/* Atom feed link in header (paired with transmit in .nav) */ +.nav { display:flex; align-items:center; gap:clamp(10px,2.2vw,20px); flex-wrap:wrap; justify-content:flex-end; } +a.header-feed-link { font-size:0.8rem; text-decoration:none; opacity:0.82; letter-spacing:0.04em; white-space:nowrap; } +a.header-feed-link:hover { opacity:1; text-decoration:underline; } +/* Full-viewport splash (theme-specific colours/animation on each .splash-THEMENAME) */ +#splash-overlay { position:fixed; inset:0; z-index:2000; display:flex; flex-direction:column; align-items:center; + justify-content:center; text-align:center; padding:max(16px,4vw); box-sizing:border-box; cursor:pointer; + transition:opacity .55s ease, visibility .55s ease, transform .55s ease; } +#splash-overlay.splash--dismissed { opacity:0 !important; visibility:hidden !important; + pointer-events:none !important; transform:scale(1.02); } +#splash-overlay:focus { outline:2px solid rgba(255,255,255,0.35); outline-offset:4px; } +/* Vignette over WebGL so 3D motion does not overpower the edges */ +#splash-overlay::before { content:""; position:absolute; inset:0; z-index:1; pointer-events:none; + background: radial-gradient(ellipse 92% 82% at 50% 42%, rgba(0,0,0,0) 32%, rgba(0,0,0,0.26) 68%, rgba(0,0,0,0.48) 100%); } +.splash-title { font-weight:700; letter-spacing:0.06em; line-height:1.15; } +.splash-tag { margin-top:0.35rem; font-size:0.76rem; letter-spacing:0.2em; text-transform:uppercase; } +.splash-hint { margin-top:1.25rem; font-size:0.72rem; letter-spacing:0.12em; } +#splash-overlay .splash-gl-canvas { position:absolute; inset:0; width:100%; height:100%; display:block; z-index:0; pointer-events:none; } +/* Frosted panel so title/tag/hint stay readable over busy shaders */ +#splash-overlay .splash-inner { position:relative; z-index:2; max-width:min(520px,92vw); + padding: clamp(1.15rem, 3.2vw, 1.75rem) clamp(1.3rem, 3.8vw, 1.95rem); border-radius:14px; + background: rgba(0, 0, 0, 0.58); backdrop-filter: blur(14px); -webkit-backdrop-filter: blur(14px); + box-shadow: 0 14px 44px rgba(0, 0, 0, 0.58), inset 0 1px 0 rgba(255, 255, 255, 0.07); } +#splash-overlay.splash-brutalist .splash-inner.splash-frame { + padding: clamp(1.4rem, 4.5vw, 2.25rem) clamp(1.1rem, 3.5vw, 1.9rem); background: rgba(0, 0, 0, 0.78); } +html.sno-splash-skip #splash-overlay { display:none !important; visibility:hidden !important; pointer-events:none !important; } </style> <div class="post-modal" id="post-modal"> <div class="modal-inner"> @@ -40,6 +104,67 @@ const navDefs = ` {{define "navscript"}} <script> + (function splashSetup() { + var el = document.getElementById('splash-overlay'); + if (!el) return; + if (document.documentElement.classList.contains('sno-splash-skip')) { + if (typeof window._snonuxSplashWebGLCleanup === 'function') { + try { window._snonuxSplashWebGLCleanup(); } catch (_) {} + window._snonuxSplashWebGLCleanup = null; + } + el.remove(); + return; + } + var splashAudioCtx = null; + var splashChimePlayed = false; + // Soft major arpeggio (G4 → C5 → E5 → G5); works once autopolicy allows audio. + function playSplashChime() { + if (splashChimePlayed) return; + try { + if (!splashAudioCtx) { + splashAudioCtx = new (window.AudioContext || window.webkitAudioContext)(); + } + var ctx = splashAudioCtx; + function ring() { + splashChimePlayed = true; + var now = ctx.currentTime; + var freqs = [392, 523.25, 659.25, 783.99]; + var i, osc, g, t0; + for (i = 0; i < freqs.length; i++) { + osc = ctx.createOscillator(); + g = ctx.createGain(); + osc.connect(g); + g.connect(ctx.destination); + osc.type = 'sine'; + osc.frequency.value = freqs[i]; + t0 = now + i * 0.075; + g.gain.setValueAtTime(0, t0); + g.gain.linearRampToValueAtTime(0.1, t0 + 0.028); + g.gain.exponentialRampToValueAtTime(0.001, t0 + 0.52); + osc.start(t0); + osc.stop(t0 + 0.55); + } + } + ctx.resume().then(ring).catch(function() {}); + } catch (_) {} + } + playSplashChime(); + el.addEventListener('pointerdown', function() { playSplashChime(); }, { passive: true }); + function dismiss() { + if (el.classList.contains('splash--dismissed')) return; + playSplashChime(); + if (typeof window._snonuxSplashWebGLCleanup === 'function') { + try { window._snonuxSplashWebGLCleanup(); } catch (_) {} + window._snonuxSplashWebGLCleanup = null; + } + el.classList.add('splash--dismissed'); + setTimeout(function() { if (el.parentNode) el.parentNode.removeChild(el); }, 600); + } + el.addEventListener('click', function(e) { e.preventDefault(); dismiss(); }); + window._snonuxDismissSplash = dismiss; + el.focus({ preventScroll: true }); + })(); + // === KEYBOARD NAVIGATION === // j / ArrowDown → next post k / ArrowUp → previous post // h / ArrowLeft → previous page l / ArrowRight → next page @@ -121,6 +246,14 @@ const navDefs = ` document.addEventListener('keydown', function(e) { if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA') return; + var splash = document.getElementById('splash-overlay'); + if (splash && !splash.classList.contains('splash--dismissed')) { + if (e.key === 'Enter' || e.key === ' ' || e.key === 'Escape') { + e.preventDefault(); + if (window._snonuxDismissSplash) window._snonuxDismissSplash(); + } + return; + } if (document.getElementById('post-modal').classList.contains('active')) { if (e.key === 'Escape') { closeModal(); e.preventDefault(); } return; diff --git a/internal/generator/theme_aurora.go b/internal/generator/theme_aurora.go index 4fa85f6..887f936 100644 --- a/internal/generator/theme_aurora.go +++ b/internal/generator/theme_aurora.go @@ -29,6 +29,8 @@ const auroraTemplate = `<!DOCTYPE html> .transmit-btn { border:1px solid var(--teal); color:var(--teal); padding:9px 20px; border-radius:20px; text-decoration:none; font-size:0.85rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--teal); color:var(--navy); } + a.header-feed-link { color:var(--green); } + a.header-feed-link:hover { color:var(--teal); text-shadow:0 0 8px var(--green); } .nav-hints { background:rgba(5,13,26,0.6); border-bottom:1px solid rgba(0,255,179,0.15); color:rgba(224,248,240,0.45); padding:5px 28px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } @@ -62,9 +64,56 @@ const auroraTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--teal); font-size:0.9rem; cursor:pointer; letter-spacing:1px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} .content{padding:14px 18px;} } + .splash-overlay.splash-aurora { + background: linear-gradient(125deg, var(--navy) 0%, #0a1a2e 40%, #0d2840 70%, var(--navy) 100%); + background-size: 200% 200%; + animation: splashAuroraShift 8s ease infinite; + } + @keyframes splashAuroraShift { 0%,100%{background-position:0% 40%} 50%{background-position:100% 60%} } + .splash-aurora .splash-ribbons { + width:min(280px,75vw); height:8px; margin:0 auto 1.25rem; border-radius:4px; + background: linear-gradient(90deg, transparent, var(--green), var(--teal), var(--purple), transparent); + opacity:0.85; animation: splashRibbonFlow 3s ease-in-out infinite alternate; + box-shadow: 0 0 24px var(--green), 0 0 40px var(--purple); + } + @keyframes splashRibbonFlow { from { transform: scaleX(0.85); opacity:0.65; } to { transform: scaleX(1); opacity:1; } } + .splash-aurora .splash-title { font-size:clamp(1.45rem,4.5vw,2rem); color:#e0f8f0; + text-shadow:0 0 20px var(--green); } + .splash-aurora .splash-tag { background:linear-gradient(90deg,var(--green),var(--teal)); + -webkit-background-clip:text; -webkit-text-fill-color:transparent; } + .splash-aurora .splash-hint { color:rgba(224,248,240,0.88); } + .splash-aurora .splash-inner { text-shadow: 0 2px 18px rgba(0,0,0,0.75); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-aurora" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-ribbons" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Aurora uplink</div> + <div class="splash-hint">Click or Enter to open the feed</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(),i,m; + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(50,1,0.1,80);ca.position.set(0,0.3,9); + var cols=[0x00ffb3,0x00cfe8,0xc084fc]; + for(i=0;i<3;i++){m=new THREE.Mesh(new THREE.TorusKnotGeometry(1.05,0.28,48,6,2,3),new THREE.MeshBasicMaterial({color:cols[i],transparent:true,opacity:0.55,wireframe:i===2}));m.rotation.y=i*2.1;m.userData.o=i;g.add(m);} + sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.y=t*0.28;g.children.forEach(function(c,ix){c.rotation.x=Math.sin(t*0.8+c.userData.o)*0.35;});ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -73,15 +122,16 @@ const auroraTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">Transmit</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← Newer</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -91,7 +141,12 @@ const auroraTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">Older →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← Newer</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">Older →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_brutalist.go b/internal/generator/theme_brutalist.go index aa0729d..2bb54da 100644 --- a/internal/generator/theme_brutalist.go +++ b/internal/generator/theme_brutalist.go @@ -30,6 +30,8 @@ const brutalistTemplate = `<!DOCTYPE html> border-radius:0; text-decoration:none; font-family:Impact; font-size:1.05rem; letter-spacing:2px; transition:all 0.1s; } .transmit-btn:hover { background:var(--red); color:#000; } + a.header-feed-link { color:#aaa; font-family:'Courier New',monospace; font-size:0.78rem; } + a.header-feed-link:hover { color:var(--red); } .nav-hints { background:#111; border-bottom:2px solid #333; color:#888; padding:5px 24px; display:flex; gap:18px; font-family:'Courier New',monospace; font-size:0.7rem; flex-wrap:wrap; } @@ -62,9 +64,46 @@ const brutalistTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--red); font-family:Impact; font-size:1.3rem; cursor:pointer; letter-spacing:2px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:10px 16px;} .logo-mark{font-size:2rem;} } + .splash-overlay.splash-brutalist { background:#000; } + .splash-brutalist .splash-frame { + border:6px solid #fff; padding:clamp(1.5rem,5vw,2.5rem) clamp(1.25rem,4vw,2rem); + box-shadow: 12px 12px 0 var(--red); animation: splashBrutalJolt 3s steps(2,end) infinite; + } + @keyframes splashBrutalJolt { 0%,100% { transform: translate(0,0); } 50% { transform: translate(2px,-2px); } } + .splash-brutalist .splash-title { font-family:Impact,sans-serif; font-size:clamp(1.8rem,6vw,2.8rem); color:#fff; } + .splash-brutalist .splash-tag { font-family:'Courier New',monospace; color:var(--red); } + .splash-brutalist .splash-hint { font-family:'Courier New',monospace; color:#c8c8c8; } + .splash-brutalist .splash-inner { text-shadow: 0 0 12px #000, 0 2px 8px #000; } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-brutalist" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner splash-frame"> + <div class="splash-title">SNONUX.FOO</div> + <div class="splash-tag">Brutalist channel</div> + <div class="splash-hint">[ CLICK OR ENTER TO TRANSMIT ]</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(50,1,0.1,80);ca.position.z=8; + var b1=new THREE.LineSegments(new THREE.EdgesGeometry(new THREE.BoxGeometry(3.4,2.4,2.4)),new THREE.LineBasicMaterial({color:0xffffff})); + var b2=new THREE.LineSegments(new THREE.EdgesGeometry(new THREE.BoxGeometry(2.2,1.6,1.6)),new THREE.LineBasicMaterial({color:0xff2200})); + b2.position.set(0.3,0.2,0.5);g.add(b1);g.add(b2);sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.x=t*0.51;g.rotation.y=t*0.73;ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -73,15 +112,16 @@ const brutalistTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>SNONUX.FOO</h1> <p class="subtitle">MICROBLOG — <a href="https://foo.zone">FOO.ZONE</a> IS THE REAL BLOG</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">TRANSMIT</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← NEWER</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -91,7 +131,12 @@ const brutalistTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">OLDER →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_glass.go b/internal/generator/theme_glass.go index 3fc951b..40e3ee3 100644 --- a/internal/generator/theme_glass.go +++ b/internal/generator/theme_glass.go @@ -30,6 +30,8 @@ const cosmosTemplate = `<!DOCTYPE html> .transmit-btn { border:1px solid var(--gold); color:var(--gold); padding:9px 20px; border-radius:20px; text-decoration:none; font-size:0.85rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--gold); color:var(--bg); } + a.header-feed-link { color:var(--blue); } + a.header-feed-link:hover { color:var(--gold); } .nav-hints { background:rgba(2,2,20,0.6); border-bottom:1px solid rgba(255,209,102,0.12); color:rgba(212,232,255,0.4); padding:5px 28px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } @@ -62,9 +64,65 @@ const cosmosTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--gold); font-size:0.9rem; cursor:pointer; letter-spacing:1px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} .content{padding:14px 18px;} } + .splash-overlay.splash-cosmos { background: radial-gradient(ellipse 100% 80% at 50% 100%, rgba(155,93,229,0.2) 0%, transparent 55%), var(--bg); } + .splash-cosmos .splash-stars { + position:absolute; inset:0; pointer-events:none; opacity:0.5; + background-image: radial-gradient(1px 1px at 20% 30%, rgba(255,255,255,0.9), transparent), + radial-gradient(1px 1px at 80% 20%, rgba(255,209,102,0.8), transparent), + radial-gradient(1px 1px at 40% 70%, rgba(76,201,240,0.7), transparent), + radial-gradient(1px 1px at 65% 55%, rgba(255,255,255,0.6), transparent); + background-size: 100% 100%; + animation: splashTwinkle 4s ease-in-out infinite alternate; + } + @keyframes splashTwinkle { from { opacity:0.35; } to { opacity:0.65; } } + .splash-cosmos .splash-inner { position:relative; z-index:1; } + .splash-cosmos .splash-orbit { + width:72px; height:72px; margin:0 auto 1rem; border:2px solid rgba(255,209,102,0.5); + border-radius:50%; animation: splashOrbitSpin 12s linear infinite; + box-shadow: 0 0 30px rgba(155,93,229,0.4); + } + @keyframes splashOrbitSpin { to { transform: rotate(360deg); } } + .splash-cosmos .splash-title { font-size:clamp(1.45rem,4.5vw,2rem); color:#d4e8ff; } + .splash-cosmos .splash-tag { + background:linear-gradient(90deg,var(--gold),var(--purple)); + -webkit-background-clip:text; -webkit-text-fill-color:transparent; } + .splash-cosmos .splash-hint { color:rgba(212,232,255,0.88); } + .splash-cosmos .splash-stars { z-index:1; } + .splash-cosmos .splash-inner { text-shadow: 0 2px 20px rgba(0,0,0,0.85); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-cosmos" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-stars" aria-hidden="true"></div> + <div class="splash-inner"> + <div class="splash-orbit" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Cosmos gate</div> + <div class="splash-hint">Engage — click or Enter</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(48,1,0.1,90);ca.position.set(0,0.5,10); + var planet=new THREE.Mesh(new THREE.SphereGeometry(1.35,24,24),new THREE.MeshBasicMaterial({color:0xc8853a,transparent:true,opacity:0.92})); + var ring=new THREE.Mesh(new THREE.TorusGeometry(2.1,0.05,8,64),new THREE.MeshBasicMaterial({color:0xffd166,transparent:true,opacity:0.85})); + ring.rotation.x=Math.PI/2.25;var moon=new THREE.Mesh(new THREE.SphereGeometry(0.35,12,12),new THREE.MeshBasicMaterial({color:0x9b5de5,transparent:true,opacity:0.8})); + moon.position.set(2.8,0.6,0.5);g.add(planet);g.add(ring);g.add(moon);sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.y=t*0.22;planet.rotation.y=t*0.35; + moon.position.x=2.6*Math.cos(t*0.7);moon.position.z=2.6*Math.sin(t*0.7);ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -73,15 +131,16 @@ const cosmosTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">Transmit</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← Newer</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -91,7 +150,12 @@ const cosmosTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">Older →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← Newer</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">Older →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_matrix.go b/internal/generator/theme_matrix.go index 8d35d7c..d964a11 100644 --- a/internal/generator/theme_matrix.go +++ b/internal/generator/theme_matrix.go @@ -40,6 +40,8 @@ const matrixTemplate = `<!DOCTYPE html> text-decoration:none; font-size:0.82rem; letter-spacing:2px; transition:all 0.1s; } .transmit-btn:hover { background:var(--g); color:var(--bg); } + a.header-feed-link { color:var(--g2); } + a.header-feed-link:hover { color:var(--g); text-shadow:0 0 8px var(--g); } .nav-hints { background:#000; border-bottom:1px solid var(--g3); color:var(--g2); padding:4px 24px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } .nav-hints kbd { background:transparent; border:1px solid var(--g3); color:var(--g); @@ -70,9 +72,58 @@ const matrixTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--g2); font-family:monospace; font-size:0.9rem; cursor:pointer; letter-spacing:2px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:10px 16px;} .content{padding:10px 16px;} } + .splash-overlay.splash-matrix { background: #000; font-family:'Courier New',monospace; } + .splash-matrix .splash-rain { + position:absolute; inset:0; overflow:hidden; pointer-events:none; opacity:0.35; z-index:1; + font-size:11px; line-height:14px; color:var(--g2); text-align:left; padding:8px; + white-space:pre; animation: splashMatrixScroll 16s linear infinite; + } + @keyframes splashMatrixScroll { to { transform: translateY(-24px); } } + .splash-matrix .splash-title { + position:relative; z-index:1; font-size:clamp(1.1rem,3.5vw,1.5rem); color:var(--g); + text-shadow:0 0 20px var(--g); letter-spacing:0.35em; + animation: splashMatrixGlow 1.8s ease-in-out infinite alternate; + } + @keyframes splashMatrixGlow { from { opacity:0.85; } to { opacity:1; text-shadow:0 0 28px var(--g); } } + .splash-matrix .splash-tag { position:relative; z-index:1; color:rgba(0,255,65,0.88); } + .splash-matrix .splash-hint { position:relative; z-index:1; color:rgba(0,255,65,0.82); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-matrix" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-rain" aria-hidden="true">01001110 01000101 01001111 +10101010 11001100 00110011 +01110011 01101110 01101111 +11001010 10100101 01011010</div> + <div class="splash-inner"> + <div class="splash-title">SNONUX.FOO</div> + <div class="splash-tag">Follow the signal</div> + <div class="splash-hint">wake up — click or enter</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,pts,pos,t0=performance.now(),N=28,i,arr; + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(55,1,0.1,80);ca.position.set(0,0.5,10); + arr=new Float32Array(N*3*20);for(i=0;i<arr.length;i+=3){arr[i]=(Math.random()-0.5)*16;arr[i+1]=Math.random()*22;arr[i+2]=(Math.random()-0.5)*8;} + var geo=new THREE.BufferGeometry();geo.setAttribute('position',new THREE.BufferAttribute(arr,3)); + pts=new THREE.Points(geo,new THREE.PointsMaterial({color:0x00ff41,size:0.14,transparent:true,opacity:0.85,blending:THREE.AdditiveBlending,depthWrite:false,sizeAttenuation:true})); + sc.add(pts);pos=geo.attributes.position;sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001,j,p; + for(j=0;j<pos.count;j++){p=j*3;pos.array[p+1]-=0.045+Math.sin(t+j*0.1)*0.012;if(pos.array[p+1]<-2)pos.array[p+1]=20;} + pos.needsUpdate=true;pts.rotation.y=t*0.15;ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -81,15 +132,16 @@ const matrixTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>SNONUX.FOO</h1> <p class="subtitle">MICROBLOG / <a href="https://foo.zone">FOO.ZONE</a> IS THE REAL BLOG</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">atom.xml</a> <a href="https://foo.zone/about" class="transmit-btn">TRANSMIT</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}"><-- NEWER</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -99,7 +151,12 @@ const matrixTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">OLDER --></a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}"><-- NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER --></a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_minimal.go b/internal/generator/theme_minimal.go index 00e9a17..4b7de26 100644 --- a/internal/generator/theme_minimal.go +++ b/internal/generator/theme_minimal.go @@ -30,6 +30,8 @@ const plasmaTemplate = `<!DOCTYPE html> .transmit-btn { border:1px solid var(--magenta); color:var(--magenta); padding:9px 20px; border-radius:20px; text-decoration:none; font-size:0.85rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--magenta); color:var(--bg); } + a.header-feed-link { color:var(--cyan); } + a.header-feed-link:hover { color:var(--magenta); } .nav-hints { background:rgba(5,0,8,0.65); border-bottom:1px solid rgba(0,240,255,0.12); color:rgba(232,224,255,0.4); padding:5px 28px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } @@ -63,9 +65,60 @@ const plasmaTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--cyan); font-size:0.9rem; cursor:pointer; letter-spacing:1px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} .content{padding:14px 18px;} } + .splash-overlay.splash-plasma { background: var(--bg); overflow:hidden; } + .splash-plasma .splash-blobs { + position:absolute; width:140%; height:140%; left:-20%; top:-20%; pointer-events:none; + background: + radial-gradient(ellipse at 30% 40%, rgba(0,240,255,0.25) 0%, transparent 45%), + radial-gradient(ellipse at 70% 60%, rgba(255,0,224,0.22) 0%, transparent 50%), + radial-gradient(ellipse at 50% 80%, rgba(255,238,0,0.12) 0%, transparent 40%); + animation: splashPlasmaDrift 10s ease-in-out infinite alternate; + filter: blur(2px); + } + @keyframes splashPlasmaDrift { + from { transform: translate(0,0) rotate(0deg); } + to { transform: translate(-4%,3%) rotate(8deg); } + } + .splash-plasma .splash-inner { position:relative; z-index:1; } + .splash-plasma .splash-title { font-size:clamp(1.45rem,4.5vw,2rem); color:#e8e0ff; + text-shadow:0 0 24px var(--cyan), 0 0 48px rgba(255,0,224,0.35); } + .splash-plasma .splash-tag { color:var(--magenta); letter-spacing:0.18em; } + .splash-plasma .splash-hint { color:rgba(232,224,255,0.86); } + .splash-plasma .splash-blobs { z-index:1; } + .splash-plasma .splash-inner { text-shadow: 0 2px 22px rgba(0,0,0,0.9); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-plasma" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-blobs" aria-hidden="true"></div> + <div class="splash-inner"> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Plasma lock</div> + <div class="splash-hint">Merge — click or Enter</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(),spec=[[0x00f0ff,1.45,0,0,0],[0xff00e0,1.05,1.2,0.4,0],[0xffee00,0.75,-1.1,-0.3,0]]; + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(50,1,0.1,60);ca.position.z=6.5; + spec.forEach(function(s){var m=new THREE.Mesh(new THREE.SphereGeometry(s[1],20,20),new THREE.MeshBasicMaterial({color:s[0],transparent:true,opacity:0.42,blending:THREE.AdditiveBlending,depthWrite:false})); + m.position.set(s[2],s[3],s[4]);m.userData.ph=s;g.add(m);}); + sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001; + g.children.forEach(function(c,ix){var ph=c.userData.ph;c.position.x=ph[2]+Math.sin(t*0.9+ix)*0.35;c.position.y=ph[3]+Math.cos(t*0.7+ix*1.3)*0.28;c.scale.setScalar(1+Math.sin(t*1.5+ix)*0.06);}); + ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -74,15 +127,16 @@ const plasmaTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">Transmit</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← Newer</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -92,7 +146,12 @@ const plasmaTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">Older →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← Newer</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">Older →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_neon.go b/internal/generator/theme_neon.go index 3197d6f..eccf558 100644 --- a/internal/generator/theme_neon.go +++ b/internal/generator/theme_neon.go @@ -27,7 +27,8 @@ const neonTemplate = `<!DOCTYPE html> .logo-title .subtitle { font-size:0.68rem; opacity:0.6; letter-spacing:1px; margin-top:2px; } .logo-title .subtitle a { color:var(--neon-cyan); text-decoration:none; } .logo-title .subtitle a:hover { text-shadow:0 0 8px var(--neon-cyan); } - .nav { display:flex; gap:16px; align-items:center; } + .nav { gap:16px; } + a.header-feed-link { color:var(--neon-cyan); text-shadow:0 0 8px rgba(0,245,255,0.35); } .transmit-btn { background:transparent; border:3px solid var(--neon-yellow); color:var(--neon-yellow); padding:12px 28px; border-radius:9999px; font-weight:600; letter-spacing:1px; display:flex; align-items:center; gap:10px; box-shadow:0 0 30px var(--neon-yellow); @@ -80,9 +81,61 @@ const neonTemplate = `<!DOCTYPE html> header { padding:14px 20px; } .transmit-btn { padding:9px 16px; font-size:0.8rem; } .nav-hints { display:none; } .modal-inner { padding:24px 16px; } } + .splash-overlay.splash-neon { + background: radial-gradient(ellipse 120% 80% at 50% 35%, rgba(0,245,255,0.14) 0%, transparent 55%), + radial-gradient(ellipse 90% 55% at 75% 85%, rgba(255,0,204,0.12) 0%, transparent 50%), + #0b001a; + } + .splash-neon .splash-deco { + width:100px; height:100px; margin:0 auto 1.25rem; border-radius:50%; + border:3px solid var(--neon-cyan); box-shadow:0 0 36px var(--neon-cyan), inset 0 0 26px rgba(0,245,255,0.15); + animation: splashNeonSpin 5s linear infinite; + } + @keyframes splashNeonSpin { to { transform: rotate(360deg); } } + .splash-neon .splash-title { + font-size: clamp(1.5rem, 5vw, 2.35rem); + animation: splashNeonPulse 2s ease-in-out infinite alternate; + } + @keyframes splashNeonPulse { + from { text-shadow: 0 0 12px var(--neon-cyan), 0 0 24px rgba(255,0,204,0.4); } + to { text-shadow: 0 0 26px var(--neon-cyan), 0 0 48px var(--neon-magenta); } + } + .splash-neon .splash-tag { color: var(--neon-yellow); } + .splash-neon .splash-hint { color: rgba(224,248,255,0.9); font-family: 'Orbitron', sans-serif; } + .splash-neon .splash-inner { text-shadow: 0 2px 24px rgba(0,0,0,0.85), 0 0 40px rgba(11,0,26,0.9); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-neon" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-deco" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Neon Nexus</div> + <div class="splash-hint">Click or Enter — establish link</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren){ren.dispose();}ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true}); + ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(52,1,0.1,100);ca.position.set(0,0.4,9); + var cols=[0x00f5ff,0xff00cc,0xffe700],i,m; + for(i=0;i<3;i++){m=new THREE.Mesh(new THREE.TorusGeometry(1.55+i*0.48,0.055,8,48),new THREE.MeshBasicMaterial({color:cols[i],transparent:true,opacity:0.92}));m.rotation.x=Math.PI/2;m.userData.sp=0.01+i*0.004;g.add(m);} + g.add(new THREE.Mesh(new THREE.SphereGeometry(0.52,20,20),new THREE.MeshBasicMaterial({color:0xffe700,transparent:true,opacity:0.95}))); + sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.y=t*0.42;g.rotation.x=Math.sin(t*0.65)*0.12;g.children.forEach(function(c){if(c.userData.sp)c.rotation.z+=c.userData.sp;});ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -123,9 +176,11 @@ const neonTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn"> <i class="fa-solid fa-feather-pointed"></i> TRANSMIT TO NEXUS </a> @@ -133,9 +188,6 @@ const neonTemplate = `<!DOCTYPE html> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}} - <div class="page-nav"><a href="{{.PrevPage}}">← NEWER TRANSMISSIONS</a></div> - {{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -145,8 +197,11 @@ const neonTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}} - <div class="page-nav"><a href="{{.NextPage}}">OLDER TRANSMISSIONS →</a></div> + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← NEWER TRANSMISSIONS</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER TRANSMISSIONS →</a>{{end}} + </div> {{end}} </div> </div> diff --git a/internal/generator/theme_ocean.go b/internal/generator/theme_ocean.go index af4c80b..76469e5 100644 --- a/internal/generator/theme_ocean.go +++ b/internal/generator/theme_ocean.go @@ -28,6 +28,8 @@ const oceanTemplate = `<!DOCTYPE html> .transmit-btn { border:1px solid var(--teal); color:var(--teal); padding:9px 20px; border-radius:20px; text-decoration:none; font-size:0.85rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--teal); color:var(--navy); } + a.header-feed-link { color:var(--aqua); } + a.header-feed-link:hover { color:var(--foam); } .nav-hints { background:rgba(3,4,94,0.65); border-bottom:1px solid rgba(0,180,216,0.18); color:rgba(202,240,248,0.45); padding:5px 28px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } @@ -60,9 +62,54 @@ const oceanTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--teal); font-size:0.9rem; cursor:pointer; letter-spacing:1px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} .content{padding:14px 18px;} } + .splash-overlay.splash-ocean { + background: linear-gradient(180deg, var(--navy) 0%, var(--deep) 45%, #001a3d 100%); + } + .splash-ocean .splash-wave { + width:min(320px,88vw); height:14px; margin:0 auto 1.2rem; border-radius:50%; + background: radial-gradient(ellipse at 50% 0%, var(--aqua), transparent 70%); + opacity:0.7; animation: splashWaveBob 2.8s ease-in-out infinite; + box-shadow: 0 8px 40px rgba(0,180,216,0.35); + } + @keyframes splashWaveBob { 0%,100%{ transform: translateY(0) scaleX(1); } 50%{ transform: translateY(-6px) scaleX(1.05); } } + .splash-ocean .splash-title { font-size:clamp(1.45rem,4.5vw,2rem); color:var(--foam); + text-shadow:0 0 18px var(--teal); } + .splash-ocean .splash-tag { color:var(--aqua); letter-spacing:0.2em; } + .splash-ocean .splash-hint { color:rgba(202,240,248,0.88); } + .splash-ocean .splash-inner { text-shadow: 0 2px 16px rgba(3,4,94,0.9); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-ocean" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-wave" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Deep channel</div> + <div class="splash-hint">Surface — click or Enter</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(),i; + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(50,1,0.1,70);ca.position.set(0,0.2,9); + for(i=0;i<5;i++){var b=new THREE.Mesh(new THREE.SphereGeometry(0.25+Math.random()*0.35,12,12),new THREE.MeshBasicMaterial({color:0x48cae4,transparent:true,opacity:0.65})); + b.position.set((Math.random()-0.5)*7,(Math.random()-0.5)*4,(Math.random()-0.5)*3);b.userData.dy=0.02+Math.random()*0.03;b.userData.x=b.position.x;b.userData.y0=b.position.y;g.add(b);} + var jelly=new THREE.Mesh(new THREE.SphereGeometry(1.1,16,16),new THREE.MeshBasicMaterial({color:0x00b4d8,transparent:true,opacity:0.35,wireframe:true})); + g.add(jelly);sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;jelly.rotation.y=t*0.4; + g.children.forEach(function(c){if(c.userData.dy){c.position.y+=Math.sin(t*2+c.userData.x)*0.008;c.position.x=c.userData.x+Math.sin(t+c.userData.y0)*0.15;}});ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -71,15 +118,16 @@ const oceanTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">Transmit</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← Newer</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -89,7 +137,12 @@ const oceanTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">Older →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← Newer</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">Older →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_paper.go b/internal/generator/theme_paper.go index 9ed40a2..02b0bb8 100644 --- a/internal/generator/theme_paper.go +++ b/internal/generator/theme_paper.go @@ -28,6 +28,8 @@ const volcanoTemplate = `<!DOCTYPE html> .transmit-btn { border:1px solid var(--lava); color:var(--lava); padding:9px 20px; border-radius:4px; text-decoration:none; font-size:0.85rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--lava); color:var(--bg); } + a.header-feed-link { color:var(--ember); } + a.header-feed-link:hover { color:var(--hot); text-shadow:0 0 8px var(--lava); } .nav-hints { background:rgba(13,8,2,0.7); border-bottom:1px solid rgba(255,68,0,0.15); color:rgba(255,232,204,0.4); padding:5px 28px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } @@ -60,9 +62,57 @@ const volcanoTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--ember); font-size:0.9rem; cursor:pointer; letter-spacing:1px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} .content{padding:14px 18px;} } + .splash-overlay.splash-volcano { + background: radial-gradient(ellipse 80% 60% at 50% 100%, rgba(255,68,0,0.35) 0%, transparent 50%), var(--bg); + } + .splash-volcano .splash-ember { + width:min(200px,55vw); height:4px; margin:0 auto 1.3rem; border-radius:2px; + background: linear-gradient(90deg, transparent, var(--lava), var(--hot), var(--ember), transparent); + animation: splashEmberPulse 1.6s ease-in-out infinite alternate; + box-shadow: 0 0 20px var(--lava), 0 6px 30px rgba(255,68,0,0.4); + } + @keyframes splashEmberPulse { from { opacity:0.6; transform: scaleX(0.9); } to { opacity:1; transform: scaleX(1); } } + .splash-volcano .splash-title { font-size:clamp(1.45rem,4.5vw,2rem); color:#ffe8cc; + text-shadow:0 0 20px var(--lava); } + .splash-volcano .splash-tag { color:var(--ember); letter-spacing:0.15em; } + .splash-volcano .splash-hint { color:rgba(255,232,204,0.88); } + .splash-volcano .splash-inner { text-shadow: 0 2px 18px rgba(0,0,0,0.85); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-volcano" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-ember" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Volcano vent</div> + <div class="splash-hint">Erupt into feed — click or Enter</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(),N=180,i,arr,geo,pts,pos; + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(52,1,0.1,70);ca.position.set(0,1.2,9); + var cone=new THREE.Mesh(new THREE.ConeGeometry(1.5,4.2,10,1,true),new THREE.MeshBasicMaterial({color:0xff4400,wireframe:true,transparent:true,opacity:0.55})); + cone.position.y=-0.8;g.add(cone); + arr=new Float32Array(N*3);for(i=0;i<N;i++){arr[i*3]=(Math.random()-0.5)*5;arr[i*3+1]=Math.random()*5;arr[i*3+2]=(Math.random()-0.5)*5;} + geo=new THREE.BufferGeometry();geo.setAttribute('position',new THREE.BufferAttribute(arr,3)); + pts=new THREE.Points(geo,new THREE.PointsMaterial({color:0xff8c00,size:0.1,transparent:true,opacity:0.75,blending:THREE.AdditiveBlending,depthWrite:false,sizeAttenuation:true})); + g.add(pts);pos=geo.attributes.position;sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001,j,p,sp; + for(j=0;j<pos.count;j++){p=j*3;sp=0.05+Math.sin(t*2.2+j*0.13)*0.018;pos.array[p+1]+=sp;if(pos.array[p+1]>6){pos.array[p+1]=-0.5;pos.array[p]=Math.sin(j*1.7+t)*2.2;}} + pos.needsUpdate=true;cone.rotation.y=t*0.25;ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -71,15 +121,16 @@ const volcanoTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">Transmit</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← Newer</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -89,7 +140,12 @@ const volcanoTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">Older →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← Newer</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">Older →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_retro.go b/internal/generator/theme_retro.go index 008fc92..563e37b 100644 --- a/internal/generator/theme_retro.go +++ b/internal/generator/theme_retro.go @@ -41,6 +41,8 @@ const retroTemplate = `<!DOCTYPE html> text-decoration:none; font-size:0.82rem; letter-spacing:2px; transition:all 0.1s; } .transmit-btn:hover { background:var(--amber); color:var(--bg); } + a.header-feed-link { color:var(--dim); } + a.header-feed-link:hover { color:var(--amber); text-shadow:0 0 6px var(--amber); } .nav-hints { background:var(--bg2); border-bottom:1px solid var(--dim); color:var(--dim); padding:4px 24px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } .nav-hints kbd { background:transparent; border:1px solid var(--dim); color:var(--amber); @@ -73,9 +75,51 @@ const retroTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--dim); font-family:monospace; font-size:0.9rem; cursor:pointer; letter-spacing:2px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:10px 16px;} .content{padding:10px 16px;} } + .splash-overlay.splash-retro { background: var(--bg); font-family:'Courier New',monospace; } + .splash-retro::after { + content:''; position:absolute; inset:0; pointer-events:none; opacity:0.35; + background: repeating-linear-gradient(0deg, transparent, transparent 2px, rgba(0,0,0,0.2) 2px, rgba(0,0,0,0.2) 4px); + } + .splash-retro .splash-inner { position:relative; z-index:1; } + .splash-retro .splash-title { + font-size:clamp(1.15rem,3.8vw,1.55rem); color:var(--amber); + text-shadow:0 0 14px var(--amber); letter-spacing:0.3em; + animation: splashRetroFlicker 4s ease-in-out infinite; + } + @keyframes splashRetroFlicker { 0%,100%{opacity:1} 50%{opacity:0.92} } + .splash-retro .splash-tag { color:#d4a020; } + .splash-retro .splash-hint { color:#c99528; } + .splash-retro .splash-inner { text-shadow: 0 0 10px #000, 0 2px 8px #000; } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-retro" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-title">*** SNONUX BBS ***</div> + <div class="splash-tag">Amber phosphor mode</div> + <div class="splash-hint">Press Enter or click to connect</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(48,1,0.1,60);ca.position.z=7.5; + var bx=new THREE.Mesh(new THREE.BoxGeometry(2.6,2.6,2.6),new THREE.MeshBasicMaterial({color:0xffb000,wireframe:true,transparent:true,opacity:0.9})); + var oc=new THREE.Mesh(new THREE.OctahedronGeometry(1.35,0),new THREE.MeshBasicMaterial({color:0xffb000,wireframe:true,transparent:true,opacity:0.55})); + g.add(bx);g.add(oc);sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.x=t*0.44;g.rotation.y=t*0.71;oc.rotation.z=t*0.9;ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -84,15 +128,16 @@ const retroTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>SNONUX.FOO</h1> <p class="subtitle">MICROBLOG / <a href="https://foo.zone">FOO.ZONE</a> IS THE REAL BLOG</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">TRANSMIT</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}"><-- NEWER</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -102,7 +147,12 @@ const retroTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">OLDER --></a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}"><-- NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER --></a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_synthwave.go b/internal/generator/theme_synthwave.go index 366c5b5..94aaf55 100644 --- a/internal/generator/theme_synthwave.go +++ b/internal/generator/theme_synthwave.go @@ -34,6 +34,7 @@ const synthwaveTemplate = `<!DOCTYPE html> border-radius:4px; text-decoration:none; letter-spacing:1px; font-size:0.88rem; transition:all 0.2s; } .transmit-btn:hover { background:var(--orange); color:var(--bg); } + a.header-feed-link { color:var(--pink); font-family:'Share Tech Mono',monospace; } .nav-hints { background:rgba(13,2,33,0.75); border-bottom:1px solid rgba(255,45,120,0.3); color:rgba(255,255,255,0.45); padding:5px 20px; display:flex; gap:18px; font-size:0.68rem; font-family:'Share Tech Mono',monospace; flex-wrap:wrap; } @@ -64,9 +65,70 @@ const synthwaveTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--orange); font-family:'Russo One',sans-serif; font-size:0.9rem; cursor:pointer; letter-spacing:2px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:12px 18px;} } + .splash-overlay.splash-synthwave { + background: linear-gradient(180deg, #2a0a3e 0%, var(--bg) 38%, #1a0630 100%); + } + .splash-synthwave .splash-grid { + position:absolute; inset:0; opacity:0.35; pointer-events:none; z-index:1; + background: linear-gradient(90deg, rgba(255,45,120,0.08) 1px, transparent 1px) 0 0 / 48px 48px, + linear-gradient(rgba(191,63,255,0.06) 1px, transparent 1px) 0 0 / 48px 48px; + transform: perspective(280px) rotateX(68deg) scale(2.2); + transform-origin: 50% 85%; + animation: splashGridDrift 10s linear infinite; + } + @keyframes splashGridDrift { to { background-position: 48px 48px, 0 96px; } } + .splash-synthwave .splash-sun { + width:min(140px,35vw); height:min(140px,35vw); margin:0 auto 1rem; border-radius:50%; + background: radial-gradient(circle, var(--orange) 0%, var(--pink) 45%, transparent 70%); + box-shadow: 0 0 60px var(--pink), 0 0 100px var(--orange); + animation: splashSunPulse 2.5s ease-in-out infinite alternate; + } + @keyframes splashSunPulse { + from { transform: scale(0.95); opacity: 0.85; } + to { transform: scale(1.05); opacity: 1; } + } + .splash-synthwave .splash-title { + font-family:'Russo One',sans-serif; font-size:clamp(1.5rem,5vw,2.2rem); + background: linear-gradient(90deg,var(--pink),var(--orange)); + -webkit-background-clip:text; -webkit-text-fill-color:transparent; + } + .splash-synthwave .splash-tag { font-family:'Share Tech Mono',monospace; color:var(--purple); } + .splash-synthwave .splash-hint { font-family:'Share Tech Mono',monospace; color:rgba(255,255,255,0.88); } + .splash-synthwave .splash-inner { text-shadow: 0 2px 20px rgba(13,2,33,0.95); } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-synthwave" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-grid" aria-hidden="true"></div> + <div class="splash-inner"> + <div class="splash-sun" aria-hidden="true"></div> + <div class="splash-title">snonux.foo</div> + <div class="splash-tag">Synthwave uplink</div> + <div class="splash-hint">Click or Enter to ride the grid</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,g=new THREE.Group(),t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(58,1,0.1,120);ca.position.set(0,1.2,7); + var sun=new THREE.Mesh(new THREE.SphereGeometry(1.35,28,28),new THREE.MeshBasicMaterial({color:0xff6b2b,transparent:true,opacity:0.95})); + sun.position.y=2.1;g.add(sun); + var gr=new THREE.Mesh(new THREE.PlaneGeometry(28,28,20,20),new THREE.MeshBasicMaterial({color:0xbf3fff,wireframe:true,transparent:true,opacity:0.4})); + gr.rotation.x=-Math.PI/2.15;gr.position.y=-2.4;g.add(gr); + sc.add(g);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;g.rotation.y=Math.sin(t*0.35)*0.08;sun.position.y=2.1+Math.sin(t*1.2)*0.08;sun.scale.setScalar(1+Math.sin(t*2)*0.04);ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -75,15 +137,16 @@ const synthwaveTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog — <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">Atom feed</a> <a href="https://foo.zone/about" class="transmit-btn">TRANSMIT TO NEXUS</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}">← NEWER</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -93,7 +156,12 @@ const synthwaveTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">OLDER →</a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}">← NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER →</a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} diff --git a/internal/generator/theme_terminal.go b/internal/generator/theme_terminal.go index 0503d79..9464664 100644 --- a/internal/generator/theme_terminal.go +++ b/internal/generator/theme_terminal.go @@ -39,6 +39,8 @@ const terminalTemplate = `<!DOCTYPE html> border-radius:0; text-decoration:none; letter-spacing:2px; font-size:0.85rem; transition:all 0.2s; } .nav a.transmit-btn:hover { background:var(--p); color:var(--bg); } + a.header-feed-link { color:var(--dim); } + a.header-feed-link:hover { color:var(--p); } .nav-hints { background:var(--bg2); border-bottom:1px solid var(--dim); color:var(--dim); padding:5px 24px; display:flex; gap:18px; font-size:0.68rem; flex-wrap:wrap; } .nav-hints kbd { background:transparent; border:1px solid var(--dim); color:var(--p); @@ -69,9 +71,45 @@ const terminalTemplate = `<!DOCTYPE html> .modal-close { float:right; background:none; border:none; color:var(--p); font-family:monospace; font-size:0.9rem; cursor:pointer; letter-spacing:2px; } @media(max-width:640px) { .nav-hints{display:none;} header{padding:10px 16px;} .content{padding:12px 16px;} } + .splash-overlay.splash-terminal { background: var(--bg); font-family:'Courier New',monospace; } + .splash-terminal .splash-prompt { text-align:left; font-size:0.9rem; color:rgba(51,255,51,0.78); margin-bottom:0.5rem; } + .splash-terminal .splash-title { font-size:clamp(1.2rem,4vw,1.65rem); color:var(--p); + text-shadow:0 0 12px var(--p); letter-spacing:0.15em; } + .splash-terminal .splash-cursor::after { content:'█'; animation: splashTermBlink 1s step-end infinite; color:var(--p); } + @keyframes splashTermBlink { 0%,100%{opacity:1} 50%{opacity:0} } + .splash-terminal .splash-tag { color:rgba(51,255,51,0.85); letter-spacing:0.25em; } + .splash-terminal .splash-hint { color:rgba(51,255,51,0.8); } + .splash-terminal .splash-inner { text-shadow: 0 0 8px #000, 0 2px 12px #000; } </style> </head> <body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-terminal" tabindex="-1" aria-label="Open microblog"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-prompt">> ./snonux --boot</div> + <div class="splash-title splash-cursor">LINK ESTABLISHED</div> + <div class="splash-tag">TERMINAL SESSION</div> + <div class="splash-hint">[ click / enter to continue ]</div> + </div> + </div> + <script> + (function(){ + if(document.documentElement.classList.contains('sno-splash-skip'))return; + var cv=document.getElementById('splash-gl-canvas'); + if(!cv||typeof THREE==='undefined')return; + var raf,ren,sc,ca,m,t0=performance.now(); + function cleanup(){window.removeEventListener('resize',sz);if(raf)cancelAnimationFrame(raf);raf=null;if(ren)ren.dispose();ren=null;window._snonuxSplashWebGLCleanup=null;} + window._snonuxSplashWebGLCleanup=cleanup; + function sz(){var w=cv.clientWidth||2,h=cv.clientHeight||2;if(ren)ren.setSize(w,h,false);if(ca){ca.aspect=w/h;ca.updateProjectionMatrix();}} + ren=new THREE.WebGLRenderer({canvas:cv,antialias:true,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(Math.min(window.devicePixelRatio||1,2)); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(48,1,0.1,60);ca.position.z=7; + m=new THREE.Mesh(new THREE.IcosahedronGeometry(2.3,1),new THREE.MeshBasicMaterial({color:0x33ff33,wireframe:true,transparent:true,opacity:0.88})); + sc.add(m);sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop);var t=(now-t0)*0.001;m.rotation.x=t*0.62;m.rotation.y=t*0.88;ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> <canvas id="three-canvas"></canvas> <div class="overlay"> <header> @@ -80,15 +118,16 @@ const terminalTemplate = `<!DOCTYPE html> <div class="logo-title"> <h1>snonux.foo</h1> <p class="subtitle">microblog / <a href="https://foo.zone">foo.zone</a> is the real blog</p> + <p class="logo-host">Site served by a Raspberry Pi 3</p> </div> </div> <div class="nav"> + <a href="atom.xml" class="header-feed-link" rel="alternate" title="Atom feed" type="application/atom+xml">atom.xml</a> <a href="https://foo.zone/about" class="transmit-btn">> TRANSMIT</a> </div> </header> {{template "navhints" .}} <div class="content" id="post-content"> - {{if .PrevPage}}<div class="page-nav"><a href="{{.PrevPage}}"><-- NEWER</a></div>{{end}} {{range $i, $post := .Posts}} <div class="post" data-index="{{$i}}" onclick="selectPost({{$i}})"> <div class="post-header"> @@ -98,7 +137,12 @@ const terminalTemplate = `<!DOCTYPE html> <div class="post-text">{{$post.ContentHTML}}</div> </div> {{end}} - {{if .NextPage}}<div class="page-nav"><a href="{{.NextPage}}">OLDER --></a></div>{{end}} + {{if or .PrevPage .NextPage}} + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}"><-- NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER --></a>{{end}} + </div> + {{end}} </div> </div> {{template "navmodal" .}} |
