From b85e5f281b195a2895a9e057047419ca2d5ef694 Mon Sep 17 00:00:00 2001 From: dangrubbb Date: Thu, 12 Jun 2025 02:25:29 +0100 Subject: [PATCH] Upload files to "/" Initial Commit --- .gitignore | 1 + index.html | 254 +++++++++++++++++++++++++++++++++++++++++++++++++++++ script.js | 129 +++++++++++++++++++++++++++ style.css | 196 +++++++++++++++++++++++++++++++++++++++++ 4 files changed, 580 insertions(+) create mode 100644 .gitignore create mode 100644 index.html create mode 100644 script.js create mode 100644 style.css diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..1b5b003 --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +deploy_dangrubb.net.sh \ No newline at end of file diff --git a/index.html b/index.html new file mode 100644 index 0000000..12e7143 --- /dev/null +++ b/index.html @@ -0,0 +1,254 @@ + + + + + + dangrubb.net + + + + + + + +
+
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+ +
+
+ +

+ dangrubb.net +

+

+ ai + blog + media + storage + cv + git

+
+
+
+ + + + + diff --git a/script.js b/script.js new file mode 100644 index 0000000..59b84a2 --- /dev/null +++ b/script.js @@ -0,0 +1,129 @@ +"use strict"; +const random = (min, max) => { + return Math.round(Math.random() * (max - min)) + min; +}; +const getKeyFrames = (name, glitchPercentageDuration, steps = 3, tick = 0.1) => { + const percentageStep = 100 / steps; + const keyframes = []; + // First keyframe + const baseKeys = [0]; + for (let i = 1; i < steps; i++) { + const p = i * percentageStep; + baseKeys.push(p); + baseKeys.push(p + glitchPercentageDuration); + } + // Last keyframe + baseKeys.push(100); + keyframes.push({ + keys: baseKeys, + css: { + transform: "none", + filter: "hue-rotate(0) drop-shadow(0 0 0 transparent)" // Hack to force animation in Safari + } + }); + for (let i = 1; i < steps; i++) { + const p = i * percentageStep; + // Blue / red shadow + const color = Math.random() > 0.5 ? "rgb(255 0 0 / 0.1)" : "rgb(0 0 255 / 0.1)"; + const shadowX = random(-4, 4); + const shadowY = random(-4, 4); + keyframes.push({ + keys: [p + tick, p + glitchPercentageDuration - tick], + css: { + transform: `translateX(var(--glitch-x-${i}))`, + filter: `hue-rotate(var(--glitch-hue-${i})) drop-shadow(${shadowX}px ${shadowY}px 0 ${color})` + } + }); + } + const css = keyframes + .map((keyframe) => { + const keys = keyframe.keys + .map((key) => `${key.toFixed(2)}%`) + .join(",\n "); + const content = Object.entries(keyframe.css) + .map(([key, value]) => ` ${key}: ${value};`) + .join("\n "); + return [keys, "{", content, "}"].join("\n "); + }) + .join("\n\n "); + return `@keyframes ${name} {\n ${css}\n}`; +}; +const getStripHTML = (top, stripHeight) => { + const duration = random(5, 10); + const name = `glitch-${duration}`; + return `
`; +}; +const getGlitchHTML = (height) => { + let i = 0; + const html = []; + while (1) { + const stripHeight = random(1, 6); + if (i + stripHeight < height) { + const strip = getStripHTML(i, stripHeight); + html.push(strip); + } + else { + // Last strip + const strip = getStripHTML(i, height - i); + html.push(strip); + break; + } + i = i + stripHeight; + } + return html; +}; +/* + +If you want to generate new CSS/HTML dinamically, +uncomment the code below. + +*/ +// const html = getGlitchHTML(62); +// // HTML +// const $glitch = document.querySelector(".bard") as HTMLElement; +// $glitch.innerHTML = html.join("\n"); +// // CSS +// const css = [5,6,7,8,9,10].map((n) => { +// const glitchDurationMS = 500; +// const glitchPercentageDuration = (glitchDurationMS * 100) / (n * 1000); +// return getKeyFrames(`glitch-${n}`, glitchPercentageDuration); +// }); +// // Add generated CSS to the page +// const $style = document.createElement("style"); +// $style.innerHTML = css.join("\n"); +// document.head.appendChild($style); +// // ----- Debug -------------- // +// // Not used for the animation // +// const $code = document.querySelector(".code"); +// const escape = (html) => { +// return html +// .replace(/&/g, "&") +// .replace(//g, ">") +// .replace(/"/g, """) +// .replace(/'/g, "'"); +// }; +// $code.innerHTML = ` +//
+//
HTML
+//
${escape(html.join("\n\n"))}
+//
+//
+//
CSS
+//
${css.join("\n\n")}
+//
+// `; \ No newline at end of file diff --git a/style.css b/style.css new file mode 100644 index 0000000..6a3b10f --- /dev/null +++ b/style.css @@ -0,0 +1,196 @@ +body { + background: #0d0a14; + padding: 40px 20px; + font-family: ui-monospace, "Cascadia Code", "Source Code Pro", Menlo, Consolas, "DejaVu Sans Mono", monospace; + color: #9d9aa4; +} + +.bard { + margin: 0 auto 40px; + width: 65em; + height: 62em; + font-size: 2px; +} +@media (min-width: 360px) { + .bard { + font-size: 3px; + } +} +@media (min-width: 600px) { + .bard { + font-size: 4px; + } +} +@media (min-width: 1000px) { + .bard { + font-size: 5px; + } +} + +.bard img { + width: 100%; + height: auto; + position: absolute; + left: 0; + top: 0; +} + +.strip { + overflow: hidden; + position: relative; + animation-iteration-count: infinite; + animation-direction: alternate; + animation-timing-function: linear; + image-rendering: pixelated; + image-rendering: -moz-crisp-edges; + image-rendering: crisp-edges; + background-size: 100% auto; + background-image: url(); +} + +@keyframes glitch-5 { + 0.00%, 33.33%, 43.33%, 66.67%, 76.67%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 43.23% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(0px -4px 0 rgba(0, 0, 255, 0.1)); + } + 66.77%, 76.57% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(-1px 0px 0 rgba(255, 0, 0, 0.1)); + } +} +@keyframes glitch-6 { + 0.00%, 33.33%, 41.67%, 66.67%, 75.00%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 41.57% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(-2px 3px 0 rgba(255, 0, 0, 0.1)); + } + 66.77%, 74.90% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(-3px -2px 0 rgba(0, 0, 255, 0.1)); + } +} +@keyframes glitch-7 { + 0.00%, 33.33%, 40.48%, 66.67%, 73.81%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 40.38% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(0px -3px 0 rgba(0, 0, 255, 0.1)); + } + 66.77%, 73.71% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(4px 1px 0 rgba(255, 0, 0, 0.1)); + } +} +@keyframes glitch-8 { + 0.00%, 33.33%, 39.58%, 66.67%, 72.92%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 39.48% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(-1px -1px 0 rgba(0, 0, 255, 0.1)); + } + 66.77%, 72.82% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(3px -1px 0 rgba(0, 0, 255, 0.1)); + } +} +@keyframes glitch-9 { + 0.00%, 33.33%, 38.89%, 66.67%, 72.22%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 38.79% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(1px -3px 0 rgba(255, 0, 0, 0.1)); + } + 66.77%, 72.12% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(1px 1px 0 rgba(255, 0, 0, 0.1)); + } +} +@keyframes glitch-10 { + 0.00%, 33.33%, 38.33%, 66.67%, 71.67%, 100.00% { + transform: none; + filter: hue-rotate(0) drop-shadow(0 0 0 transparent); + } + 33.43%, 38.23% { + transform: translateX(var(--glitch-x-1)); + filter: hue-rotate(var(--glitch-hue-1)) drop-shadow(3px -1px 0 rgba(255, 0, 0, 0.1)); + } + 66.77%, 71.57% { + transform: translateX(var(--glitch-x-2)); + filter: hue-rotate(var(--glitch-hue-2)) drop-shadow(-3px 2px 0 rgba(255, 0, 0, 0.1)); + } +} +p { + margin: 20px auto 0; + position: relative; + max-width: 400px; + text-align: center; + line-height: 1.3; + font-size: 0.875rem; +} + +a { + color: #6be1e9; +} + +a:focus, +a:hover, +a:visited { + opacity: 0.8; +} + +a:focus-visible { + outline: 2px solid currentColor; + outline-offset: 2px; +} + +.debug { + margin: 40px auto 0; + position: relative; + max-width: 800px; + display: flex; + flex-direction: column; + align-items: center; + max-width: 100%; +} + +.code { + margin: 10px auto 0; +} +@media (min-width: 600px) { + .code { + display: flex; + gap: 32px; + justify-content: center; + align-items: flex-start; + } +} + +.heading { + margin: 0 0 10px; + font-size: 1.5rem; + font-weight: bold; +} + +.column { + border: 1px solid rgba(255, 255, 255, 0.1); + padding: 10px; + border-radius: 4px; + color: #e8c57d; +} + +.column + .column { + color: #aa95bd; +} \ No newline at end of file