diff options
| author | Paul Buetow <paul@buetow.org> | 2026-04-16 09:00:39 +0300 |
|---|---|---|
| committer | Paul Buetow <paul@buetow.org> | 2026-04-16 09:00:39 +0300 |
| commit | a846354700471a48b3bc885b343928460059d9d4 (patch) | |
| tree | 1bf79294eb58506e8b7c4f5670e55c2e2bf3b332 | |
| parent | 24b04a11f215d898cf95bcc6c0316b3d25e2b35f (diff) | |
v0.3.0: add DOS theme with VT323 webfontv0.3.0
Amp-Thread-ID: https://ampcode.com/threads/T-019d94cc-99a9-74af-8f3d-9521cd73324f
Co-authored-by: Amp <amp@ampcode.com>
| -rw-r--r-- | internal/generator/theme_dos.go | 242 | ||||
| -rw-r--r-- | internal/generator/theme_sounds.go | 11 | ||||
| -rw-r--r-- | internal/generator/themes.go | 1 | ||||
| -rw-r--r-- | internal/version/version.go | 2 |
4 files changed, 255 insertions, 1 deletions
diff --git a/internal/generator/theme_dos.go b/internal/generator/theme_dos.go new file mode 100644 index 0000000..521f6ed --- /dev/null +++ b/internal/generator/theme_dos.go @@ -0,0 +1,242 @@ +package generator + +// dosTemplate is a classic DOS / IBM PC text-mode theme — blue background +// (#0000aa), white/yellow text, VT323 webfont for authentic VGA bitmap look, +// double-line box-drawing borders, and a BIOS-style layout. +// WebGL scene: falling green "rain" characters (BASIC-era) on the blue BG. +const dosTemplate = `<!DOCTYPE html> +<html lang="en"> +<head> + <meta charset="UTF-8"> + <meta name="viewport" content="width=device-width, initial-scale=1.0"> + <title>SNONUX.FOO - DOS</title> + <link rel="preconnect" href="https://fonts.googleapis.com"> + <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> + <link href="https://fonts.googleapis.com/css2?family=VT323&display=swap" rel="stylesheet"> + <script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/r134/three.min.js"></script> + <style> + :root { --dos-blue:#0000aa; --dos-lblue:#5555ff; --dos-white:#aaaaaa; + --dos-bwhite:#ffffff; --dos-yellow:#ffff55; --dos-cyan:#55ffff; + --dos-red:#ff5555; --dos-bg:#000088; --dos-black:#000000; } + * { margin:0; padding:0; box-sizing:border-box; } + body { font-family:'VT323','Courier New',monospace; background:var(--dos-blue); + color:var(--dos-bwhite); overflow:hidden; height:100vh; font-size:18px; } + #three-canvas { position:fixed; top:0; left:0; width:100%; height:100%; z-index:1; } + .overlay { position:relative; z-index:10; height:100vh; display:flex; flex-direction:column; } + header { padding:8px 20px; background:var(--dos-white); color:var(--dos-blue); + display:flex; align-items:center; justify-content:space-between; } + .logo { display:flex; align-items:center; gap:12px; } + .logo-mark { font-size:1.8rem; color:var(--dos-blue); font-weight:bold; } + .logo-title h1 { font-size:1.4rem; color:var(--dos-blue); font-weight:normal; letter-spacing:2px; } + .logo-title .subtitle { font-size:0.85rem; color:var(--dos-blue); margin-top:1px; } + .logo-title .subtitle a { color:var(--dos-blue); text-decoration:underline; } + .logo-title .subtitle a:hover { color:var(--dos-black); } + .transmit-btn { border:2px solid var(--dos-blue); color:var(--dos-blue); padding:4px 14px; + text-decoration:none; font-size:1rem; letter-spacing:1px; + transition:all 0.1s; } + .transmit-btn:hover { background:var(--dos-blue); color:var(--dos-bwhite); } + a.header-feed-link { color:var(--dos-blue); } + a.header-feed-link:hover { color:var(--dos-black); } + .nav-hints { background:var(--dos-blue); border-bottom:2px solid var(--dos-lblue); + color:var(--dos-cyan); padding:4px 20px; display:flex; gap:16px; + font-size:0.85rem; flex-wrap:wrap; } + .nav-hints kbd { background:var(--dos-black); border:1px solid var(--dos-lblue); + color:var(--dos-yellow); padding:0 5px; margin:0 2px; } + .content { flex:1; overflow-y:auto; padding:12px 20px; + scrollbar-width:thin; scrollbar-color:var(--dos-lblue) var(--dos-blue); } + .page-nav { display:flex; justify-content:center; margin:10px 0; } + .page-nav a { border:2px solid var(--dos-lblue); color:var(--dos-yellow); padding:6px 18px; + text-decoration:none; font-size:1rem; letter-spacing:1px; } + .page-nav a:hover { background:var(--dos-lblue); color:var(--dos-bwhite); } + .page-nav-footer { flex-shrink:0; padding:6px 20px; display:flex; justify-content:center; + background:var(--dos-white); color:var(--dos-blue); } + .page-nav-footer .page-nav a { border-color:var(--dos-blue); color:var(--dos-blue); } + .page-nav-footer .page-nav a:hover { background:var(--dos-blue); color:var(--dos-bwhite); } + .post { background:var(--dos-black); border:2px solid var(--dos-lblue); + padding:12px 14px; margin-bottom:8px; cursor:pointer; + transition:border-color 0.1s; } + .post:hover { border-color:var(--dos-yellow); + box-shadow:0 0 0 1px var(--dos-yellow); } + .post-active { border-color:var(--dos-yellow) !important; + background:rgba(0,0,170,0.3) !important; + box-shadow:0 0 0 2px var(--dos-yellow),inset 3px 0 0 var(--dos-yellow) !important; } + .post-header { display:flex; justify-content:space-between; margin-bottom:8px; font-size:1rem; } + .post-header strong { color:var(--dos-yellow); } + .post-time { color:var(--dos-cyan); font-size:0.95rem; } + .post-text { line-height:1.5; font-size:1.05rem; } + .post-text a { color:var(--dos-cyan); text-decoration:underline; } + .post-text a:hover { color:var(--dos-yellow); } + .post-image { max-width:100%; margin-top:8px; border:2px solid var(--dos-lblue); } + .post-audio { width:100%; margin-top:8px; } + .post-modal { display:none; position:fixed; inset:0; z-index:100; + background:rgba(0,0,0,0.95); overflow-y:auto; padding:40px 20px; } + .post-modal.active { display:block; } + .modal-inner { max-width:740px; margin:0 auto; background:var(--dos-black); + border:2px solid var(--dos-yellow); padding:24px; + box-shadow:0 0 20px rgba(85,85,255,0.4); } + .modal-close { float:right; background:var(--dos-white); border:2px outset var(--dos-bwhite); + color:var(--dos-blue); font-family:'VT323','Courier New',monospace; + font-size:1rem; cursor:pointer; padding:2px 8px; } + .modal-close:hover { background:var(--dos-blue); color:var(--dos-bwhite); + border-style:inset; } + @media(max-width:640px) { .nav-hints{display:none;} header{padding:6px 12px;} .content{padding:8px 12px;} } + .splash-overlay.splash-dos { background:var(--dos-blue); font-family:'VT323','Courier New',monospace; } + .splash-dos .splash-inner { position:relative; z-index:1; } + .splash-dos .splash-title { + font-size:clamp(1.4rem,4.5vw,2rem); color:var(--dos-bwhite); + letter-spacing:0.15em; + animation: splashDosBlink 1s step-end infinite; + } + @keyframes splashDosBlink { 0%,100%{border-right:0.6em solid var(--dos-bwhite)} 50%{border-right:0.6em solid transparent} } + .splash-dos .splash-tag { color:var(--dos-yellow); letter-spacing:0.15em; } + .splash-dos .splash-hint { color:var(--dos-cyan); } + .splash-dos .splash-inner { + background:var(--dos-black); border:2px solid var(--dos-lblue); + text-shadow:none; box-shadow:4px 4px 0 rgba(0,0,0,0.5); + } +{{template "navSharedCSSInner"}} + </style> +</head> +<body> + {{template "splashGate"}} + <div id="splash-overlay" class="splash-overlay splash-dos" role="dialog" aria-modal="true" aria-label="Open microblog" tabindex="-1"> + <canvas class="splash-gl-canvas" id="splash-gl-canvas" aria-hidden="true"></canvas> + <div class="splash-inner"> + <div class="splash-title">C:\SNONUX></div> + <div class="splash-tag">MS-DOS v6.22</div> + <div class="splash-hint">Press any key 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,drops=[],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:false,alpha:true});ren.setClearColor(0,0);ren.setPixelRatio(1); + sc=new THREE.Scene();ca=new THREE.PerspectiveCamera(50,1,0.1,80);ca.position.z=20; + var geo=new THREE.PlaneGeometry(0.22,0.32); + for(var i=0;i<60;i++){ + var mat=new THREE.MeshBasicMaterial({color:0x55ff55,transparent:true,opacity:0.3+Math.random()*0.4}); + var m=new THREE.Mesh(geo,mat); + m.position.set((Math.random()-0.5)*28, Math.random()*22-11, (Math.random()-0.5)*5); + m.userData.speed=0.5+Math.random()*1.5; + sc.add(m); drops.push(m); + } + sz();window.addEventListener('resize',sz); + function loop(now){raf=requestAnimationFrame(loop); + for(var i=0;i<drops.length;i++){ + drops[i].position.y-=drops[i].userData.speed*0.06; + if(drops[i].position.y<-12) drops[i].position.y=12; + } + ren.render(sc,ca);} + raf=requestAnimationFrame(loop); + })(); + </script> + <canvas id="three-canvas"></canvas> + <div class="overlay"> + <header> + <div class="logo"> + <span class="logo-mark">C:\></span> + <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">Served by NetBSD on 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">ABOUT</a> + </div> + </header> + {{template "navhints" .}} + <div class="content" id="post-content"> + {{range $i, $post := .Posts}} + <div class="post" id="post-{{$post.ID}}" data-index="{{$i}}"> + <div class="post-header"> + <div><strong>@SNONUX</strong></div> + <div class="post-time">{{$post.FormattedTime}}</div> + </div> + <div class="post-text">{{$post.ContentHTML}}</div> + </div> + {{end}} + </div> + {{if or .PrevPage .NextPage}} + <footer class="page-nav-footer" aria-label="Pagination"> + <div class="page-nav page-nav-dual"> + {{if .PrevPage}}<a href="{{.PrevPage}}"><-- NEWER</a>{{end}} + {{if .NextPage}}<a href="{{.NextPage}}">OLDER --></a>{{end}} + </div> + </footer> + {{end}} + </div> + {{template "navmodal" .}} + <script> + (function() { + var scene, camera, renderer, clock; + var columns = []; + + function initThree() { + scene = new THREE.Scene(); + scene.background = new THREE.Color(0x000088); + + camera = new THREE.PerspectiveCamera(60, window.innerWidth/window.innerHeight, 0.1, 200); + camera.position.set(0, 0, 40); + + renderer = new THREE.WebGLRenderer({ canvas: document.getElementById('three-canvas'), antialias: false }); + renderer.setSize(window.innerWidth, window.innerHeight); + renderer.setPixelRatio(1); + clock = new THREE.Clock(); + + var geo = new THREE.PlaneGeometry(0.35, 0.5); + + for (var c = 0; c < 30; c++) { + var col = []; + var x = (c - 15) * 2.2; + var speed = 1.5 + Math.random() * 3; + var startY = Math.random() * 60 - 30; + for (var r = 0; r < 8; r++) { + var brightness = 1.0 - (r / 8) * 0.7; + var color = new THREE.Color(brightness * 0.33, brightness, brightness * 0.33); + var mat = new THREE.MeshBasicMaterial({ color: color, transparent: true, opacity: brightness * 0.5 }); + var mesh = new THREE.Mesh(geo, mat); + mesh.position.set(x, startY - r * 0.7, 0); + scene.add(mesh); + col.push({ mesh: mesh, offset: r * 0.7 }); + } + columns.push({ chars: col, x: x, speed: speed, y: startY }); + } + + window.addEventListener('resize', onResize); + animate(); + } + + function onResize() { + camera.aspect = window.innerWidth / window.innerHeight; + camera.updateProjectionMatrix(); + renderer.setSize(window.innerWidth, window.innerHeight); + } + + function animate() { + requestAnimationFrame(animate); + var t = clock.getElapsedTime(); + for (var c = 0; c < columns.length; c++) { + var col = columns[c]; + var y = col.y - t * col.speed; + y = ((y % 60) + 60) % 60 - 30; + for (var r = 0; r < col.chars.length; r++) { + col.chars[r].mesh.position.y = y - col.chars[r].offset; + } + } + renderer.render(scene, camera); + } + + initThree(); + })(); + </script> + {{template "navscript" .}} +</body> +</html>` diff --git a/internal/generator/theme_sounds.go b/internal/generator/theme_sounds.go index e41e7a5..4876ab1 100644 --- a/internal/generator/theme_sounds.go +++ b/internal/generator/theme_sounds.go @@ -47,6 +47,7 @@ var themeSoundPresets = map[string]themeSounds{ "aurora": soundsAurora(), "matrix": soundsMatrix(), "ocean": soundsOcean(), + "dos": soundsDos(), "retro": soundsRetro(), "cosmos": soundsCosmos(), } @@ -141,6 +142,16 @@ func soundsOcean() themeSounds { return s } +func soundsDos() themeSounds { + var s themeSounds + s.Splash.Freqs = []float64{800, 1000} + s.Splash.Spacing, s.Splash.Gain, s.Splash.Wave = 0.08, 0.12, "square" + s.Nav.Freq, s.Nav.Wave, s.Nav.Dur, s.Nav.Gain = 1000, "square", 0.03, 0.1 + s.Open.Wave, s.Open.Start, s.Open.End, s.Open.Dur, s.Open.Gain = "square", 400, 800, 0.1, 0.1 + s.Close.Wave, s.Close.Start, s.Close.End, s.Close.Dur, s.Close.Gain = "square", 800, 200, 0.1, 0.09 + return s +} + func soundsRetro() themeSounds { var s themeSounds s.Splash.Freqs = []float64{1046.5, 1318.5} diff --git a/internal/generator/themes.go b/internal/generator/themes.go index 4d35db6..47afaef 100644 --- a/internal/generator/themes.go +++ b/internal/generator/themes.go @@ -14,6 +14,7 @@ var themeRegistry = map[string]string{ "aurora": auroraTemplate, "matrix": matrixTemplate, "ocean": oceanTemplate, + "dos": dosTemplate, "retro": retroTemplate, "cosmos": cosmosTemplate, // replaced "glass" — ringed planet, nebula, asteroids } diff --git a/internal/version/version.go b/internal/version/version.go index 4e2512d..742745d 100644 --- a/internal/version/version.go +++ b/internal/version/version.go @@ -2,4 +2,4 @@ package version // Version is the application version (semantic versioning). -const Version = "0.2.0" +const Version = "0.3.0" |
