mirror of
https://github.com/wah-su/wah-su.github.io.git
synced 2025-04-06 08:14:40 +00:00
feat: add Footer
This commit is contained in:
parent
ee8e03f2ab
commit
714dd79c46
8 changed files with 253 additions and 5 deletions
|
@ -1,5 +1,11 @@
|
|||
@import "tailwindcss";
|
||||
|
||||
.ubuntu-regular {
|
||||
font-family: "Ubuntu", serif;
|
||||
font-weight: 400;
|
||||
font-style: normal;
|
||||
}
|
||||
|
||||
.ubuntu-mono-regular {
|
||||
font-family: "Ubuntu Mono", serif;
|
||||
font-weight: 400;
|
||||
|
@ -11,4 +17,18 @@
|
|||
font-optical-sizing: auto;
|
||||
font-weight: 400;
|
||||
font-style: normal;
|
||||
}
|
||||
|
||||
body {
|
||||
position: relative;
|
||||
|
||||
/* Create the dots */
|
||||
background-image:
|
||||
radial-gradient(circle at 1px 1px, #33333366 1px, transparent 0),
|
||||
linear-gradient(to right, #33333366 1px, transparent 1px),
|
||||
linear-gradient(to bottom, #33333366 1px, transparent 1px);
|
||||
|
||||
background-size: 20px 20px;
|
||||
background-position: 0 0, 0 0, 0 0;
|
||||
background-attachment: fixed;
|
||||
}
|
|
@ -518,6 +518,9 @@
|
|||
}
|
||||
}
|
||||
@layer utilities {
|
||||
.invisible {
|
||||
visibility: hidden;
|
||||
}
|
||||
.absolute {
|
||||
position: absolute;
|
||||
}
|
||||
|
@ -530,6 +533,12 @@
|
|||
.inset-0 {
|
||||
inset: calc(var(--spacing) * 0);
|
||||
}
|
||||
.top-0 {
|
||||
top: calc(var(--spacing) * 0);
|
||||
}
|
||||
.left-0 {
|
||||
left: calc(var(--spacing) * 0);
|
||||
}
|
||||
.-z-10 {
|
||||
z-index: calc(10 * -1);
|
||||
}
|
||||
|
@ -584,6 +593,9 @@
|
|||
.mx-auto {
|
||||
margin-inline: auto;
|
||||
}
|
||||
.ml-1 {
|
||||
margin-left: calc(var(--spacing) * 1);
|
||||
}
|
||||
.ml-auto {
|
||||
margin-left: auto;
|
||||
}
|
||||
|
@ -596,12 +608,21 @@
|
|||
.grid {
|
||||
display: grid;
|
||||
}
|
||||
.inline {
|
||||
display: inline;
|
||||
}
|
||||
.aspect-square {
|
||||
aspect-ratio: 1 / 1;
|
||||
}
|
||||
.aspect-video {
|
||||
aspect-ratio: var(--aspect-video);
|
||||
}
|
||||
.h-4 {
|
||||
height: calc(var(--spacing) * 4);
|
||||
}
|
||||
.h-6 {
|
||||
height: calc(var(--spacing) * 6);
|
||||
}
|
||||
.h-8 {
|
||||
height: calc(var(--spacing) * 8);
|
||||
}
|
||||
|
@ -620,6 +641,12 @@
|
|||
.min-h-\[438px\] {
|
||||
min-height: 438px;
|
||||
}
|
||||
.w-4 {
|
||||
width: calc(var(--spacing) * 4);
|
||||
}
|
||||
.w-6 {
|
||||
width: calc(var(--spacing) * 6);
|
||||
}
|
||||
.w-\[114px\] {
|
||||
width: 114px;
|
||||
}
|
||||
|
@ -641,6 +668,18 @@
|
|||
.flex-shrink-0 {
|
||||
flex-shrink: 0;
|
||||
}
|
||||
.scale-80 {
|
||||
--tw-scale-x: 80%;
|
||||
--tw-scale-y: 80%;
|
||||
--tw-scale-z: 80%;
|
||||
scale: var(--tw-scale-x) var(--tw-scale-y);
|
||||
}
|
||||
.scale-90 {
|
||||
--tw-scale-x: 90%;
|
||||
--tw-scale-y: 90%;
|
||||
--tw-scale-z: 90%;
|
||||
scale: var(--tw-scale-x) var(--tw-scale-y);
|
||||
}
|
||||
.scale-140 {
|
||||
--tw-scale-x: 140%;
|
||||
--tw-scale-y: 140%;
|
||||
|
@ -653,6 +692,9 @@
|
|||
--tw-scale-z: 150%;
|
||||
scale: var(--tw-scale-x) var(--tw-scale-y);
|
||||
}
|
||||
.animate-ping {
|
||||
animation: var(--animate-ping);
|
||||
}
|
||||
.grid-cols-1 {
|
||||
grid-template-columns: repeat(1, minmax(0, 1fr));
|
||||
}
|
||||
|
@ -671,12 +713,18 @@
|
|||
.flex-col-reverse {
|
||||
flex-direction: column-reverse;
|
||||
}
|
||||
.flex-row {
|
||||
flex-direction: row;
|
||||
}
|
||||
.items-center {
|
||||
align-items: center;
|
||||
}
|
||||
.items-end {
|
||||
align-items: flex-end;
|
||||
}
|
||||
.items-start {
|
||||
align-items: flex-start;
|
||||
}
|
||||
.justify-center {
|
||||
justify-content: center;
|
||||
}
|
||||
|
@ -689,6 +737,9 @@
|
|||
.gap-2 {
|
||||
gap: calc(var(--spacing) * 2);
|
||||
}
|
||||
.gap-3 {
|
||||
gap: calc(var(--spacing) * 3);
|
||||
}
|
||||
.gap-4 {
|
||||
gap: calc(var(--spacing) * 4);
|
||||
}
|
||||
|
@ -701,6 +752,9 @@
|
|||
.overflow-x-hidden {
|
||||
overflow-x: hidden;
|
||||
}
|
||||
.rounded-full {
|
||||
border-radius: calc(infinity * 1px);
|
||||
}
|
||||
.rounded-xl {
|
||||
border-radius: var(--radius-xl);
|
||||
}
|
||||
|
@ -714,12 +768,18 @@
|
|||
.bg-\[\#1A0F05\] {
|
||||
background-color: #1A0F05;
|
||||
}
|
||||
.bg-\[\#3B0D25\] {
|
||||
background-color: #3B0D25;
|
||||
.bg-\[\#3b0d25\] {
|
||||
background-color: #3b0d25;
|
||||
}
|
||||
.bg-\[var\(--color\)\] {
|
||||
background-color: var(--color);
|
||||
}
|
||||
.bg-\[var\(--ping-color\)\] {
|
||||
background-color: var(--ping-color);
|
||||
}
|
||||
.bg-white {
|
||||
background-color: var(--color-white);
|
||||
}
|
||||
.bg-gradient-to-r {
|
||||
--tw-gradient-position: to right in oklab;
|
||||
background-image: linear-gradient(var(--tw-gradient-stops));
|
||||
|
@ -738,6 +798,9 @@
|
|||
.to-\[75\%\] {
|
||||
--tw-gradient-to-position: 75%;
|
||||
}
|
||||
.object-contain {
|
||||
object-fit: contain;
|
||||
}
|
||||
.object-cover {
|
||||
object-fit: cover;
|
||||
}
|
||||
|
@ -792,11 +855,24 @@
|
|||
.text-white {
|
||||
color: var(--color-white);
|
||||
}
|
||||
.transition {
|
||||
transition-property: color, background-color, border-color, outline-color, text-decoration-color, fill, stroke, --tw-gradient-from, --tw-gradient-via, --tw-gradient-to, opacity, box-shadow, transform, translate, scale, rotate, filter, -webkit-backdrop-filter, backdrop-filter;
|
||||
transition-timing-function: var(--tw-ease, var(--default-transition-timing-function));
|
||||
transition-duration: var(--tw-duration, var(--default-transition-duration));
|
||||
}
|
||||
.transition-colors {
|
||||
transition-property: color, background-color, border-color, outline-color, text-decoration-color, fill, stroke, --tw-gradient-from, --tw-gradient-via, --tw-gradient-to;
|
||||
transition-timing-function: var(--tw-ease, var(--default-transition-timing-function));
|
||||
transition-duration: var(--tw-duration, var(--default-transition-duration));
|
||||
}
|
||||
.duration-400 {
|
||||
--tw-duration: 400ms;
|
||||
transition-duration: 400ms;
|
||||
}
|
||||
.ease-in {
|
||||
--tw-ease: var(--ease-in);
|
||||
transition-timing-function: var(--ease-in);
|
||||
}
|
||||
.hover\:border-\[\#FF851A\] {
|
||||
&:hover {
|
||||
@media (hover: hover) {
|
||||
|
@ -883,6 +959,11 @@
|
|||
}
|
||||
}
|
||||
}
|
||||
.ubuntu-regular {
|
||||
font-family: "Ubuntu", serif;
|
||||
font-weight: 400;
|
||||
font-style: normal;
|
||||
}
|
||||
.ubuntu-mono-regular {
|
||||
font-family: "Ubuntu Mono", serif;
|
||||
font-weight: 400;
|
||||
|
@ -894,6 +975,13 @@
|
|||
font-weight: 400;
|
||||
font-style: normal;
|
||||
}
|
||||
body {
|
||||
position: relative;
|
||||
background-image: radial-gradient(circle at 1px 1px, #33333366 1px, transparent 0), linear-gradient(to right, #33333366 1px, transparent 1px), linear-gradient(to bottom, #33333366 1px, transparent 1px);
|
||||
background-size: 20px 20px;
|
||||
background-position: 0 0, 0 0, 0 0;
|
||||
background-attachment: fixed;
|
||||
}
|
||||
@keyframes spin {
|
||||
to {
|
||||
transform: rotate(360deg);
|
||||
|
@ -990,3 +1078,11 @@
|
|||
syntax: "*";
|
||||
inherits: false;
|
||||
}
|
||||
@property --tw-duration {
|
||||
syntax: "*";
|
||||
inherits: false;
|
||||
}
|
||||
@property --tw-ease {
|
||||
syntax: "*";
|
||||
inherits: false;
|
||||
}
|
||||
|
|
BIN
src/static/images/GitHub_Invertocat_Dark.png
Normal file
BIN
src/static/images/GitHub_Invertocat_Dark.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 9.1 KiB |
BIN
src/static/images/Unsplash_Symbol.png
Normal file
BIN
src/static/images/Unsplash_Symbol.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.4 KiB |
BIN
src/static/images/avatar_48.jpg
Normal file
BIN
src/static/images/avatar_48.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.7 KiB |
75
src/static/js/checkstatus.js
Normal file
75
src/static/js/checkstatus.js
Normal file
|
@ -0,0 +1,75 @@
|
|||
let statusIcon = null;
|
||||
let statusIconPing = null;
|
||||
let statusText = null;
|
||||
|
||||
const serviceUp = {
|
||||
color: "#05df72",
|
||||
text: "All Services Operational",
|
||||
};
|
||||
const serviceDegraded = {
|
||||
color: "#fcc800",
|
||||
text: "Degraded Services",
|
||||
};
|
||||
const serviceDown = {
|
||||
color: "#c10007",
|
||||
text: "All Services Down",
|
||||
};
|
||||
const serviceUnknown = {
|
||||
color: "#6a7282",
|
||||
text: "Unknown",
|
||||
};
|
||||
|
||||
async function getServicesHealth() {
|
||||
try {
|
||||
const serviceStatus = await fetch(
|
||||
"https://status.wah.su/api/status-page/heartbeat/services"
|
||||
);
|
||||
|
||||
const services = await serviceStatus.json();
|
||||
const heartbeatDict = services.heartbeatList;
|
||||
let lastHeartbeats = [];
|
||||
|
||||
for (const [key, value] of Object.entries(heartbeatDict)) {
|
||||
lastHeartbeats.push(
|
||||
heartbeatDict[key][heartbeatDict[key].length - 1].status
|
||||
);
|
||||
}
|
||||
const count = lastHeartbeats.reduce((partialSum, a) => partialSum + a, 0);
|
||||
|
||||
switch (count) {
|
||||
case lastHeartbeats.length: {
|
||||
statusIcon.style.setProperty("--ping-color", serviceUp.color);
|
||||
statusIconPing.classList.remove("invisible");
|
||||
statusText.textContent = serviceUp.text;
|
||||
break;
|
||||
}
|
||||
case 0: {
|
||||
statusIcon.style.setProperty("--ping-color", serviceDown.color);
|
||||
statusIconPing.classList.add("invisible");
|
||||
statusText.textContent = serviceDown.text;
|
||||
break;
|
||||
}
|
||||
default: {
|
||||
statusIcon.style.setProperty("--ping-color", serviceDegraded.color);
|
||||
statusIconPing.classList.remove("invisible");
|
||||
statusText.textContent = serviceDegraded.text;
|
||||
break;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
statusIcon.style.setProperty("--ping-color", serviceUnknown.color);
|
||||
statusIconPing.classList.add("invisible");
|
||||
statusText.textContent = serviceUnknown.text;
|
||||
console.log("Failed to fetch services status: " + error);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
window.onload = () => {
|
||||
statusIcon = document.getElementById("status-icon");
|
||||
statusIconPing = document.getElementById("status-icon-ping");
|
||||
statusText = document.getElementById("status-text");
|
||||
|
||||
getServicesHealth();
|
||||
setInterval(getServicesHealth, 600000);
|
||||
};
|
|
@ -17,6 +17,7 @@ export default function Base({ children, isDev }: BaseProps) {
|
|||
/>
|
||||
<title>WAH.su</title>
|
||||
{isDev ? <script src="/static/js/hotreload.js"></script> : ""}
|
||||
<script src="/static/js/checkstatus.js"></script>
|
||||
<link rel="stylesheet" href="/static/css/tailwind.css" />
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com"></link>
|
||||
<link
|
||||
|
@ -25,7 +26,7 @@ export default function Base({ children, isDev }: BaseProps) {
|
|||
crossOrigin="anonymous"
|
||||
></link>
|
||||
<link
|
||||
href="https://fonts.googleapis.com/css2?family=Sometype+Mono:ital,wght@0,400..700;1,400..700&family=Ubuntu+Mono:ital,wght@0,400;0,700;1,400;1,700&display=swap"
|
||||
href="https://fonts.googleapis.com/css2?family=Sometype+Mono&family=Ubuntu&family=Ubuntu+Mono&display=swap"
|
||||
rel="stylesheet"
|
||||
></link>
|
||||
</head>
|
||||
|
|
|
@ -5,7 +5,63 @@ interface FooterProps {
|
|||
export default function Footer({ className }: FooterProps) {
|
||||
return (
|
||||
<div
|
||||
className={`bg-[#3B0D25] w-full h-full ${className} rounded-xl p-8 flex items-center justify-center gap-8`}
|
||||
></div>
|
||||
className={`bg-[#3b0d25] w-full h-full ${className} rounded-xl p-8 flex flex-col items-center justify-center ubuntu-regular`}
|
||||
>
|
||||
<div className="flex items-start flex-col gap-4">
|
||||
<a
|
||||
href="https://status.wah.su"
|
||||
className="flex flex-row items-center justify-center gap-3"
|
||||
target="_blank"
|
||||
>
|
||||
<div
|
||||
style={{ "--ping-color": "#6a7282" } as React.CSSProperties}
|
||||
className="ml-1 w-4 h-4 relative transition ease-in bg-[var(--ping-color)] rounded-full duration-400"
|
||||
id="status-icon"
|
||||
>
|
||||
<div
|
||||
id="status-icon-ping"
|
||||
className="invisible absolute w-4 h-4 top-0 left-0 scale-90 animate-ping transition ease-in bg-[var(--ping-color)] rounded-full duration-400"
|
||||
></div>
|
||||
</div>
|
||||
<p id="status-text">...</p>
|
||||
</a>
|
||||
<div className="flex gap-8">
|
||||
<div className="flex gap-2">
|
||||
<div className="inline h-6 w-6 bg-white rounded-full overflow-hidden">
|
||||
<img
|
||||
src="/static/images/Unsplash_Symbol.png"
|
||||
className="w-full h-full object-contain"
|
||||
alt=""
|
||||
/>
|
||||
</div>
|
||||
<span>Photo by</span>{" "}
|
||||
<a href="https://unsplash.com/@lgtts">Ilse Orsel</a> <span>on</span>{" "}
|
||||
<a href="https://unsplash.com">Unsplash</a>
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex gap-8">
|
||||
<a href="https://wah.su/radiquum" className="flex gap-2">
|
||||
<div className="h-6 w-6 bg-white rounded-full overflow-hidden">
|
||||
<img
|
||||
src="/static/images/avatar_48.jpg"
|
||||
className="w-full h-full object-contain"
|
||||
alt=""
|
||||
/>
|
||||
</div>
|
||||
<p>Run by @Radiquum</p>
|
||||
</a>
|
||||
<a href="https://github.com/wah-su" className="flex gap-2">
|
||||
<div className="h-6 w-6 bg-white rounded-full">
|
||||
<img
|
||||
src="/static/images/GitHub_Invertocat_Dark.png"
|
||||
className="w-full h-full scale-80 object-contain"
|
||||
alt=""
|
||||
/>
|
||||
</div>
|
||||
<p>Find us on GitHub</p>
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue