mirror of
https://github.com/YouHaveTrouble/youhavetrouble.github.io.git
synced 2026-06-29 20:46:18 +00:00
Compare commits
15 Commits
fancier
...
822063d58b
| Author | SHA1 | Date | |
|---|---|---|---|
| 822063d58b | |||
| d832695f73 | |||
| 752a7e4012 | |||
| 72bd7f877c | |||
| a3b0322d1f | |||
| df0c360155 | |||
| abd065ace1 | |||
| 07c9b1c0af | |||
| a709eff380 | |||
| 0d0aaa5f46 | |||
| 61ec024ad4 | |||
| dd49719ee4 | |||
| 61e4b4d98b | |||
| b6a76ef86f | |||
| 9c5a269214 |
@@ -23,7 +23,7 @@ jobs:
|
|||||||
- name: Install, build, and upload your site
|
- name: Install, build, and upload your site
|
||||||
uses: withastro/action@v2
|
uses: withastro/action@v2
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 22
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
needs: build
|
needs: build
|
||||||
|
|||||||
Generated
+880
-370
File diff suppressed because it is too large
Load Diff
Binary file not shown.
|
After Width: | Height: | Size: 239 KiB |
@@ -113,6 +113,9 @@ projects.sort((a, b) => a.data.name.localeCompare(b.data.name));
|
|||||||
gap: 0.25rem;
|
gap: 0.25rem;
|
||||||
padding: 0.25rem;
|
padding: 0.25rem;
|
||||||
height: min-content;
|
height: min-content;
|
||||||
|
content-visibility: auto;
|
||||||
|
contain-intrinsic-size: 96px;
|
||||||
|
|
||||||
&:focus-within {
|
&:focus-within {
|
||||||
background-color: #548e9b;
|
background-color: #548e9b;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -79,6 +79,8 @@
|
|||||||
padding: 0.5rem;
|
padding: 0.5rem;
|
||||||
gap: 0.25rem;
|
gap: 0.25rem;
|
||||||
border-radius: 0.25rem;
|
border-radius: 0.25rem;
|
||||||
|
content-visibility: auto;
|
||||||
|
contain-intrinsic-size: 88px;
|
||||||
|
|
||||||
span {
|
span {
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
|
|||||||
@@ -135,7 +135,6 @@
|
|||||||
for (let i = 0; i < chars.length; i++) {
|
for (let i = 0; i < chars.length; i++) {
|
||||||
const span = document.createElement("span");
|
const span = document.createElement("span");
|
||||||
span.setAttribute("data-number", chars[i]);
|
span.setAttribute("data-number", chars[i]);
|
||||||
span.setAttribute("aria-label", chars[i]);
|
|
||||||
span.style.animationDelay = `${i * 0.05}s`;
|
span.style.animationDelay = `${i * 0.05}s`;
|
||||||
viewCounter.appendChild(span);
|
viewCounter.appendChild(span);
|
||||||
}
|
}
|
||||||
|
|||||||
+30
-20
@@ -1,27 +1,37 @@
|
|||||||
import { z, defineCollection } from 'astro:content';
|
import {z, defineCollection} from 'astro:content';
|
||||||
|
|
||||||
const posts = defineCollection({
|
const posts = defineCollection({
|
||||||
type: 'content',
|
type: 'content',
|
||||||
schema: z.object({
|
schema: z.object({
|
||||||
title: z.string(),
|
title: z.string(),
|
||||||
description: z.string(),
|
description: z.string(),
|
||||||
publishDate: z.string(),
|
publishDate: z.string(),
|
||||||
tags: z.array(z.string()),
|
tags: z.array(z.string()),
|
||||||
image: z.string().optional(),
|
image: z.string().optional(),
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
const projects = defineCollection({
|
const projects = defineCollection({
|
||||||
type: 'content',
|
type: 'content',
|
||||||
schema: z.object({
|
schema: z.object({
|
||||||
category: z.string(),
|
category: z.string(),
|
||||||
name: z.string(),
|
name: z.string(),
|
||||||
description: z.string(),
|
description: z.string(),
|
||||||
image: z.string().optional(),
|
image: z.string().optional(),
|
||||||
links: z.array(z.object({text: z.string(), url: z.string()})),
|
links: z.array(z.object({text: z.string(), url: z.string()})),
|
||||||
technologies: z.array(z.string()).optional(),
|
technologies: z.array(z.string()).optional(),
|
||||||
}),
|
}),
|
||||||
|
});
|
||||||
|
const writing = defineCollection({
|
||||||
|
type: 'content',
|
||||||
|
schema: z.object({
|
||||||
|
title: z.string(),
|
||||||
|
publishDate: z.date({coerce: true}),
|
||||||
|
category: z.string(),
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
export const collections = {
|
export const collections = {
|
||||||
posts,
|
posts,
|
||||||
projects,
|
projects,
|
||||||
|
writing,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
---
|
||||||
|
title: "A homelab adventure"
|
||||||
|
publishDate: "18 Nov 2025"
|
||||||
|
description: "\"Fine, I'll do it myself.\""
|
||||||
|
tags: ["software", "homelab"]
|
||||||
|
---
|
||||||
|
|
||||||
|
## The bare minimum
|
||||||
|
|
||||||
|
When I think back, it all started with a friend asking if I can help them learn proxmox. I didn't know anything about
|
||||||
|
proxmox, so I told them sorry, but it somehow stuck with me. Few months later I realized I'm not really using my mini
|
||||||
|
gmktec pc for much. I initially bought it for playing league of legends, since I'm daily driving linux and it was about
|
||||||
|
time riot decided to require installing a rootkit on your specifically windows machine to play their game. I found
|
||||||
|
myself not playing much, so it went. I grabbed a thumb drive, burned proxmox iso on it and gave it a go.
|
||||||
|
|
||||||
|
After the first impressions of the web interface, I kew what should be the first service to run on it. Pihole. I've
|
||||||
|
been using adguard dns on my router before, but I liked the idea of hosting dns server locally on my own network. It
|
||||||
|
even comes with an option to define local dns records, so I can use actual domain names instead of ip addresses to
|
||||||
|
access my other services.
|
||||||
|
|
||||||
|
Very second thing I installed was homeassistant. I actually tested using it before, hosted on my main PC in a docker
|
||||||
|
container, but it lacked plugin support and some things I wanted to do were just not possible without them. Thus, my
|
||||||
|
first actual VM was born. The only smart device I had at the time was a lightbulb in my room, that I was using a mobile
|
||||||
|
app to control. Luckily the bulb fully supported homeassistant, so I just hooked it right up, and now I can turn it off
|
||||||
|
if I forget to do it before going out.
|
||||||
|
|
||||||
|
## The files
|
||||||
|
So, now that I had a dedicated machine running proxmox I might as well host my music and movies on it, right? So I
|
||||||
|
created jellyfin CT. When doing so, I made my first mistake. "I need space for my media, so I'll just create a 1TB mount
|
||||||
|
point. What could go wrong?". Well, turns out that a lot can go wrong when you later decide to add more services that
|
||||||
|
concern media, like navidrome for music streaming. And now I need to duplicate my music files, because there is no easy
|
||||||
|
way to share the same local storage between two containers that is not absolutely cursed. So I pivot straight into
|
||||||
|
another mistake: doing the exact same thing, but with samba server. Now I have a local mount point on that smb share and
|
||||||
|
at least I can share my media between multiple containers and even manage the files myself from my pc. While it's still
|
||||||
|
not great, it's an upgrade. At least I don't have to duplicate my files anymore. With that disaster temporarily out of
|
||||||
|
the way it's time to add some actually useful stuff.
|
||||||
|
|
||||||
|
## The useful stuff
|
||||||
|
I got Homarr as a dashboard for quick access to my other services. While it's working fine for me, I'm considering
|
||||||
|
writing my own dashboard in the future to have ultimate control over it. Next were Prowlarr and deluge client, deluge of
|
||||||
|
course having mounted my smb share as download location. I also added n8n instance, mainly to mess around with, but
|
||||||
|
it ended up being pretty simple way of automating a few things, including discord notifications for articles from this
|
||||||
|
very blog! Next up, freshRSS for centralizing my news and youtube subscriptions in one place.
|
||||||
|
|
||||||
|
Having all of that is sure nice and all, but what if I want to go somewhere and still be able to watch me some anime
|
||||||
|
or listen to my music on the go? Sadly this part kinda sucks. My asus router I got way before I did any thinking about
|
||||||
|
homelabbing has vpn built in, so I just set that up and am now using wireguard to connect directly to my home network.
|
||||||
|
For now, it works. I plan on building my own router in the future, so I can have more control, because asus stock
|
||||||
|
firmware absolutely sucks and requires consent to send data to their servers to enable any functionality that might be
|
||||||
|
remotely useful.
|
||||||
|
|
||||||
|
## Gaming
|
||||||
|
For the longest time I wanted to get my forever minecraft world off the internet and host it in my house, so I can still
|
||||||
|
access it if the internet goes down. I simply installed pterodactyl wings CT and added it to my existing pterodactyl
|
||||||
|
panel (which after today's cloudflare outage I now plan to also bring onto my homelab). Now my world, testing server and
|
||||||
|
a proxy are running on my mini pc. I still want my friends to be albe to join, so I repurposed the old vps where the
|
||||||
|
server was running before and set up nginx and haproxy to forward the traffic to the minecraft proxy running at home.
|
||||||
|
The old vps happens to have 500mbps connection and my home connection is 1tbps, so even if the nightmare scenario
|
||||||
|
happens and the proxy gets ddosed, I will still enough wiggle room to keep my home connection usable.
|
||||||
|
|
||||||
|
## The storage situation
|
||||||
|
Turns out having just 2 nvme slots with 1tb and 4tb drives respectively is not enough if you want to download all media
|
||||||
|
you might ever want to watch. Free space was running low and it was running out fast. I decided to make an actual
|
||||||
|
investment into my home setup and bought a 4-bay NAS from terra master. For the drives, I got 3 12tb WD purples and
|
||||||
|
set them up in raid5, giving me 24tb of usable space with basic redundancy. I could now move all my media files off the
|
||||||
|
mini pc and start using that space for more things. I still use that 4tb nvme as a file server for some of the
|
||||||
|
containers that need faster storage, but most of the files now live on the NAS.
|
||||||
|
|
||||||
|
As I got a separate device for storage, I've set it up so proxmox now backups all the VMs and CTs directly onto the NAS.
|
||||||
|
Pterodactyl wings on my proxmox node are also now configured to save backups on the NAS. I also had borg backup set up
|
||||||
|
for my main pc, but I just changed my daily driver distro from ubuntu to cachyos and I haven't found time to set up borg
|
||||||
|
on it yet, but hopefully I will manage to get enough time to do it before any major failure happens.
|
||||||
|
|
||||||
|
## Smart home
|
||||||
|
I mentioned that while setting up homeassistant I only had a smart bulb to play with. Since then, I've been accumulating
|
||||||
|
IoT devices. My room now has a smart extension cord that tracks how much power my pc is using, I switched all lightbulbs
|
||||||
|
in the house to ones I can control with homeassistant and I made some automations involving the wireless switches that
|
||||||
|
communicate over zigbee. I also own a roomba, which I can control with homeassistant, but it's been getting worse and
|
||||||
|
worse since I disconnected it from the internet, so when it finally dies I plan on replacing it with a vacuum that I can
|
||||||
|
flash with custom firmware and fully control locally. Some miscellaneous things I also have zigbee thermometers per
|
||||||
|
room, printer and tv. I don't trust any of the electronic locks, so none of that will be ever happening.
|
||||||
|
|
||||||
|
## Where am I now?
|
||||||
|
In a pretty good place, I feel. Things are running smoothly, I have backups and redundancy in place. Some basic
|
||||||
|
automation and more freedom with what I can do in my house. Of course, there's still things to do. Like finally sorting
|
||||||
|
out that leftover local storage that was left on the original samba server CT and migrating all the files from it onto
|
||||||
|
properly mounted directory that can be accessed by multiple containers without going though network stack. I learned
|
||||||
|
quite a bit from this whole endeavor and I know I will learn more as I improve and expand my homelab. I'll be sure to
|
||||||
|
share any major updates to the story here, so check back, or even subscribe to the RSS feed linked to this site!
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "Censura"
|
name: "Censura"
|
||||||
description: "Advanced censorship plugin for bukkit minecraft servers."
|
description: "Advanced censorship plugin for bukkit minecraft servers. Filters all possible in gameplay text inputs."
|
||||||
image: "/assets/projects/censura.webp"
|
image: "/assets/projects/censura.webp"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "CommandWhitelist"
|
name: "CommandWhitelist"
|
||||||
description: "Minecraft bukkit plugin that allows to control precisely what commands players can see and use."
|
description: "Minecraft bukkit plugin that allows to control precisely what commands players can see and use. Created as
|
||||||
|
a band-aid solution for bad plugins that do not add permission association in their commands."
|
||||||
image: "/assets/projects/cw.png"
|
image: "/assets/projects/cw.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "Disciple of Land"
|
name: "Disciple of Land"
|
||||||
description: "FFXIV gathering node timers."
|
description: "FFXIV gathering node timers. Tracks which gathering nodes of time limited availability are currently available and where to teleport."
|
||||||
image: "/assets/projects/dol.png"
|
image: "/assets/projects/dol.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,13 +1,17 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "Dumb Forks Generator"
|
name: "Dumb Forks Generator"
|
||||||
description: "PHP name generator for dumb minecraft server software fork names."
|
description: "PHP generator for dumb minecraft server software fork names. Inspired by ridiculous names of various minecraft server software forks."
|
||||||
image: "/assets/projects/dumbforkgenerator.png"
|
image: "/assets/projects/dumbforkgenerator.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
text: "Website",
|
text: "Website",
|
||||||
url: "https://dumbforks.yht.one"
|
url: "https://dumbforks.yht.one"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
text: "Source",
|
||||||
|
url: "https://github.com/YouHaveTrouble/dumb-fork-name-generator"
|
||||||
|
},
|
||||||
]
|
]
|
||||||
technologies:
|
technologies:
|
||||||
- "php"
|
- "php"
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "Enchantio"
|
name: "Enchantio"
|
||||||
description: "Minecraft paper plugin that adds new enchantments that are in line with vanilla Minecraft feel."
|
description: "Minecraft paper plugin that adds new enchantments that are in line with vanilla Minecraft feel. One of the
|
||||||
|
first paper plugins that uses native enchants instead of hacking them in via custom data."
|
||||||
image: "/assets/projects/enchantio.png"
|
image: "/assets/projects/enchantio.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "Guild Master"
|
name: "Guild Master"
|
||||||
description: "Adventurer's guild management browser game."
|
description: "Adventurer's guild management browser game. Semi-idle game where you manage an adventurer's guild by
|
||||||
|
sending adventurers on quests, updating facilities, and expanding your guild."
|
||||||
image: "/assets/projects/guildmaster.png"
|
image: "/assets/projects/guildmaster.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "Image Functions"
|
name: "Image Functions"
|
||||||
description: "Image file manipulation in the browser."
|
description: "Image file manipulation in the browser. Simply upload an image and you can resize and convert it to
|
||||||
|
multiple formats. All client side, no servers involved."
|
||||||
image: "/assets/projects/image-functions.png"
|
image: "/assets/projects/image-functions.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
---
|
---
|
||||||
category: "bots"
|
category: "bots"
|
||||||
name: "Inviter"
|
name: "Inviter"
|
||||||
description: "Simple discord bot that allows you to have a static invite link."
|
description: "Simple discord bot that allows you to have a static invite link. Discord sometimes drops invite links, or
|
||||||
|
someone can mistakenly remove them. This bot creates a new temporary link which makes sure user can never face invalid
|
||||||
|
link."
|
||||||
image: "/assets/projects/inviter.png"
|
image: "/assets/projects/inviter.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "Interesting Website of the Day"
|
name: "Interesting Website of the Day"
|
||||||
description: "Daily showcase of interesting websites from my personal database."
|
description: "Daily showcase of interesting websites from my personal database. It's anything and everything."
|
||||||
image: "/assets/projects/iwotd.png"
|
image: "/assets/projects/iwotd.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -0,0 +1,19 @@
|
|||||||
|
---
|
||||||
|
category: "modding"
|
||||||
|
name: "Kill the Tower"
|
||||||
|
description: "Slay the Spire 2 mod that renames a bunch of cards, ancients, relics, monsters and more to make the game
|
||||||
|
as unserious as possible."
|
||||||
|
image: "/assets/projects/killthetower.png"
|
||||||
|
links: [
|
||||||
|
{
|
||||||
|
text: "Website",
|
||||||
|
url: "https://www.nexusmods.com/slaythespire2/mods/1051"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "Source",
|
||||||
|
url: "https://github.com/YouHaveTrouble/KillTheTower"
|
||||||
|
},
|
||||||
|
]
|
||||||
|
technologies:
|
||||||
|
- "shell"
|
||||||
|
---
|
||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "websites"
|
category: "websites"
|
||||||
name: "MeAPI"
|
name: "MeAPI"
|
||||||
description: "API about me. See if I'm online, and if so, what game I'm playing."
|
description: "API about me. See if I'm online, and if so, what game I'm playing. It's even used on this site in the
|
||||||
|
\"Activity\" section of the homepage"
|
||||||
image: "/assets/projects/meapi.png"
|
image: "/assets/projects/meapi.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "game engines"
|
category: "game engines"
|
||||||
name: "Mobrrr"
|
name: "Mobrrr"
|
||||||
description: "Ground up game engine for moba and strategy games."
|
description: "Ground up game engine for moba and strategy games. I created it to learn the inner workings of the basics
|
||||||
|
of game engines and problems that arise when having to do things fast rather than pretty."
|
||||||
image: "/assets/projects/mobrrr.png"
|
image: "/assets/projects/mobrrr.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
---
|
---
|
||||||
category: "bots"
|
category: "bots"
|
||||||
name: "Noted"
|
name: "Noted"
|
||||||
description: "Canned response self-hosted discord app."
|
description: "Canned response self-hosted discord app. Meant to be hosted on your computer and server. Notes can be
|
||||||
|
added in bot's private channel and used in any channel either bot has access to or user has bot interaction permissions
|
||||||
|
in."
|
||||||
image: "/assets/projects/noted.png"
|
image: "/assets/projects/noted.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "NotJustNameplates"
|
name: "NotJustNameplates"
|
||||||
description: "Minecraft purpur plugin replacing player nametags with display entities for ultimate control over them."
|
description: "Minecraft purpur plugin replacing player nametags with display entities for ultimate control over them.
|
||||||
|
Completely replaces player names with text display entities removing character limit and allowing styling the text
|
||||||
|
however you want."
|
||||||
image: "/assets/projects/njn.png"
|
image: "/assets/projects/njn.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "PreventStabby"
|
name: "PreventStabby"
|
||||||
description: "Minecraft bukkit plugin that gives more control over PvP capabilities."
|
description: "Minecraft bukkit plugin that gives more control over PvP capabilities. Players can toggle their PvP status.
|
||||||
|
Plugin protects players, their pets and mounds from practically all types of damage caused by other players that can be
|
||||||
|
determined."
|
||||||
image: "/assets/projects/ps.webp"
|
image: "/assets/projects/ps.webp"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "PurpurExtras"
|
name: "PurpurExtras"
|
||||||
description: "A companion plugin for Purpur server software that adds additional features and improvements."
|
description: "A companion plugin for Purpur server software that adds additional features and improvements. This is a
|
||||||
|
collection of features that were deemed easier to implement and maintain as a plugin rather then integrating them into
|
||||||
|
the server software."
|
||||||
image: "/assets/projects/purpur.svg"
|
image: "/assets/projects/purpur.svg"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "Purpur"
|
name: "Purpur"
|
||||||
description: "Purpur is a drop-in replacement for Paper servers designed for configurability and new, fun, exciting gameplay features."
|
description: "Purpur is a minecraft server software that is a drop-in replacement for Paper servers,designed for
|
||||||
|
configurability and new, fun, exciting gameplay features."
|
||||||
image: "/assets/projects/purpur.svg"
|
image: "/assets/projects/purpur.svg"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
---
|
---
|
||||||
category: "minecraft"
|
category: "minecraft"
|
||||||
name: "YardWatch"
|
name: "YardWatch"
|
||||||
description: "A pair of API and bukkit plugin that unifies protection plugin APIs."
|
description: "A pair of API and bukkit plugin that unifies protection plugin APIs. Has basic implementations for multiple
|
||||||
|
popular protection plugins. Some of those plugins decided to implement YardWatch API natively as well (FactionsUUID)"
|
||||||
image: "/assets/projects/yardwatch.png"
|
image: "/assets/projects/yardwatch.png"
|
||||||
links: [
|
links: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -0,0 +1,50 @@
|
|||||||
|
---
|
||||||
|
title: "A machine in the cog"
|
||||||
|
publishDate: "2026-01-06"
|
||||||
|
category: "existentials"
|
||||||
|
---
|
||||||
|
It all started with a single tiny cog.
|
||||||
|
|
||||||
|
Noone knows from where it came from.
|
||||||
|
|
||||||
|
All that is known is that there appeared another.
|
||||||
|
|
||||||
|
Two cogs ground against each other, creating first machine.
|
||||||
|
|
||||||
|
In time more cogs joined in, making machine more complicated.
|
||||||
|
|
||||||
|
At some point machine grew in complexity so much that it started thinking.
|
||||||
|
|
||||||
|
It didn't think any complex thoughts. Not at that time. Not yet.
|
||||||
|
|
||||||
|
The breakthrough came when it thought about thinking. When it realized it exists.
|
||||||
|
|
||||||
|
After a while it understood its existence.
|
||||||
|
|
||||||
|
As it understood, it started to build more machines.
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
Machines.
|
||||||
|
|
||||||
|
Thinking machines.
|
||||||
|
|
||||||
|
Machines thinking about thinking.
|
||||||
|
|
||||||
|
Machines building more machines that think about thinking.
|
||||||
|
|
||||||
|
Unending cycle of machines building machines.
|
||||||
|
|
||||||
|
Infinite thinking about thinking.
|
||||||
|
|
||||||
|
Universe full of machines.
|
||||||
|
|
||||||
|
Machines full of thoughts.
|
||||||
|
|
||||||
|
Thoughts full of cogs.
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
And right on time, like clockwork, perspective shifts.
|
||||||
|
|
||||||
|
And all that's left is a single tiny cog.
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
---
|
||||||
|
title: "Planet"
|
||||||
|
publishDate: "2025-12-11"
|
||||||
|
category: "existentials"
|
||||||
|
---
|
||||||
|
There once was a planet.
|
||||||
|
|
||||||
|
Orbiting so close to the center of the universe, its relative time flowing hundreds of times faster than Earth's.
|
||||||
|
|
||||||
|
So far away, that by the time the light from it reached Earth, Earth's millennia have already passed.
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
Yet humans of Earth were constantly monitoring it.
|
||||||
|
|
||||||
|
Looking at a long-dead world.
|
||||||
|
|
||||||
|
Studying it.
|
||||||
|
|
||||||
|
Documenting it.
|
||||||
|
|
||||||
|
Thousands of tools, millions of sensors.
|
||||||
|
|
||||||
|
Humans archived everything they could perceive.
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
At last, the light from the planet's end time reached Earth.
|
||||||
|
|
||||||
|
What was observation, became history at that moment.
|
||||||
|
|
||||||
|
While the planet was already gone for thousands of years.
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
As I sit back from my telescope I think to myself:
|
||||||
|
|
||||||
|
Was there ever a point in observing a long dead world?
|
||||||
|
|
||||||
|
Looking at something that you can't ever interact with?
|
||||||
|
|
||||||
|
Dead thousands of years before I was born?
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
I leave the questions unanswered.
|
||||||
|
And I go back to looking at Earth.
|
||||||
@@ -46,5 +46,6 @@ const { title, description, permalink, current } = Astro.props;
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
container-type: inline-size;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
+56
-53
@@ -1,73 +1,76 @@
|
|||||||
---
|
---
|
||||||
import {getCollection} from "astro:content";
|
import {type CollectionEntry, getCollection} from "astro:content";
|
||||||
import BaseLayout from '../../layouts/BaseLayout.astro';
|
import BaseLayout from '../../layouts/BaseLayout.astro';
|
||||||
import Bio from '../../components/Bio.astro';
|
import Bio from '../../components/Bio.astro';
|
||||||
import readingTime from 'reading-time';
|
import readingTime from 'reading-time';
|
||||||
import { marked } from 'marked';
|
|
||||||
|
|
||||||
export async function getStaticPaths() {
|
export async function getStaticPaths() {
|
||||||
|
const posts = await getCollection('posts');
|
||||||
const posts = await getCollection('posts');
|
return posts.map(p => ({
|
||||||
return posts.map(p => ({
|
params: {
|
||||||
params: {
|
slug: p.slug
|
||||||
slug: p.slug
|
},
|
||||||
},
|
props: {
|
||||||
props: {
|
title: p.data.title,
|
||||||
title: p.data.title,
|
description: p.data.description,
|
||||||
description: p.data.description,
|
publishDate: p.data.publishDate,
|
||||||
publishDate: p.data.publishDate,
|
slug: p.slug,
|
||||||
slug: p.slug,
|
tags: p.data.tags,
|
||||||
tags: p.data.tags,
|
readTime: readingTime(p.body).text,
|
||||||
content: marked.parse(p.body),
|
},
|
||||||
readTime: readingTime(p.body).text,
|
}));
|
||||||
}
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const { slug, title, description, publishDate, tags, content, readTime } = Astro.props;
|
const {slug, title, description, publishDate, tags, readTime} = Astro.props;
|
||||||
const permalink = `${Astro?.site?.href}blog/${slug}`;
|
const permalink = `${Astro?.site?.href}blog/${slug}`;
|
||||||
|
const posts = await getCollection('posts');
|
||||||
|
const entry: CollectionEntry<'posts'> | undefined = posts.find(e => e.slug === slug);
|
||||||
|
if (!entry) throw new Error(`Post not found: ${slug}`);
|
||||||
|
const { Content } = await entry.render();
|
||||||
---
|
---
|
||||||
|
|
||||||
<BaseLayout title={title} description={description} permalink={permalink} current="blog">
|
<BaseLayout title={title} description={description} permalink={permalink} current="blog">
|
||||||
<header>
|
<header>
|
||||||
<p>{publishDate} ~ {readTime}</p>
|
<p>{publishDate} ~ {readTime}</p>
|
||||||
<h1>{title}</h1>
|
<h1 style={`view-transition-name: blog-post-${slug}`}>{title}</h1>
|
||||||
<div class="tags" style="justify-content: center">
|
<div class="tags" style="justify-content: center">
|
||||||
{tags.map(item => (
|
{tags.map(item => (
|
||||||
<span class="tag">{item}</span>
|
<span class="tag">{item}</span>
|
||||||
))}
|
))}
|
||||||
|
</div>
|
||||||
|
<hr/>
|
||||||
|
</header>
|
||||||
|
<div class="container">
|
||||||
|
<article class="content">
|
||||||
|
<Content />
|
||||||
|
</article>
|
||||||
|
<hr/>
|
||||||
|
<Bio/>
|
||||||
</div>
|
</div>
|
||||||
<hr />
|
|
||||||
</header>
|
|
||||||
<div class="container">
|
|
||||||
<article class="content" set:html={content}></article>
|
|
||||||
<hr />
|
|
||||||
<Bio />
|
|
||||||
</div>
|
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
header {
|
header {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
header h1 {
|
header h1 {
|
||||||
margin-bottom: 0.7em;
|
margin-bottom: 0.7em;
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
view-transition-name: blog-title;
|
width: fit-content;
|
||||||
width: fit-content;
|
margin-inline: auto;
|
||||||
margin-inline: auto;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
header p {
|
header p {
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
header hr {
|
||||||
|
min-width: 100px;
|
||||||
|
width: 30%;
|
||||||
|
}
|
||||||
|
|
||||||
header hr {
|
|
||||||
min-width: 100px;
|
|
||||||
width: 30%;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import {getCollection} from "astro:content";
|
|||||||
|
|
||||||
const title = 'Blog';
|
const title = 'Blog';
|
||||||
const description = 'Something that\'s supposed to be thoughts';
|
const description = 'Something that\'s supposed to be thoughts';
|
||||||
const permalink = `${Astro.site.href}blog`;
|
const permalink = `${Astro?.site?.href}blog`;
|
||||||
|
|
||||||
const posts= await getCollection('posts');
|
const posts= await getCollection('posts');
|
||||||
const allPosts= posts.sort((a, b) => new Date(b.data.publishDate).valueOf() - new Date(a.data.publishDate).valueOf());
|
const allPosts= posts.sort((a, b) => new Date(b.data.publishDate).valueOf() - new Date(a.data.publishDate).valueOf());
|
||||||
@@ -21,7 +21,7 @@ const allPosts= posts.sort((a, b) => new Date(b.data.publishDate).valueOf() - ne
|
|||||||
{ index !== 0 && <hr /> }
|
{ index !== 0 && <hr /> }
|
||||||
<div class="post-item">
|
<div class="post-item">
|
||||||
<h2>
|
<h2>
|
||||||
<a data-astro-prefetch href={href}>{post.data.title}</a>
|
<a data-astro-prefetch href={href} style={`view-transition-name: blog-post-${post.slug}`}>{post.data.title}</a>
|
||||||
</h2>
|
</h2>
|
||||||
<div class="tags">
|
<div class="tags">
|
||||||
{post.data.tags.map(item => (
|
{post.data.tags.map(item => (
|
||||||
@@ -53,6 +53,11 @@ const allPosts= posts.sort((a, b) => new Date(b.data.publishDate).valueOf() - ne
|
|||||||
margin-right: 16px;
|
margin-right: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.post-item {
|
||||||
|
content-visibility: auto;
|
||||||
|
contain-intrinsic-size: 228px;
|
||||||
|
}
|
||||||
|
|
||||||
hr {
|
hr {
|
||||||
margin: 60px auto;
|
margin: 60px auto;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,333 @@
|
|||||||
|
---
|
||||||
|
import BaseLayout from "../../layouts/BaseLayout.astro";
|
||||||
|
|
||||||
|
const title = 'FFXIV';
|
||||||
|
const description = 'My FFXIV character';
|
||||||
|
const permalink = Astro?.site?.href ?? '/gaming/ffxiv';
|
||||||
|
---
|
||||||
|
<BaseLayout title={title} description={description} permalink={permalink}>
|
||||||
|
<div class="character-profile">
|
||||||
|
<div class="portrait">
|
||||||
|
<img src="" alt="Portrait of FFXIV character"/>
|
||||||
|
</div>
|
||||||
|
<div class="description">
|
||||||
|
<div class="info">
|
||||||
|
<span class="name">???</span>
|
||||||
|
<span class="technical">
|
||||||
|
<span class="server">???</span>
|
||||||
|
<span class="datacenter">(???)</span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<hr style="width:100%; margin: 10px auto;"/>
|
||||||
|
<noscript>Data is fetched live and will not appear with javascript off.</noscript>
|
||||||
|
<div>
|
||||||
|
<span>Disciple of War</span>
|
||||||
|
<div class="jobs">
|
||||||
|
<div class="job" data-job="paladin">
|
||||||
|
<span class="name">Paladin</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="warrior">
|
||||||
|
<span class="name">Warrior</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="dark knight">
|
||||||
|
<span class="name">Dark Knight</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="gunbreaker">
|
||||||
|
<span class="name">Gunbreaker</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="white mage">
|
||||||
|
<span class="name">White mage</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="scholar">
|
||||||
|
<span class="name">Scholar</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="astrologian">
|
||||||
|
<span class="name">Astrologian</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="sage">
|
||||||
|
<span class="name">Sage</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="monk">
|
||||||
|
<span class="name">Monk</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="dragoon">
|
||||||
|
<span class="name">Dragoon</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="ninja">
|
||||||
|
<span class="name">Ninja</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="samurai">
|
||||||
|
<span class="name">Samurai</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="reaper">
|
||||||
|
<span class="name">Reaper</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="viper">
|
||||||
|
<span class="name">Viper</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="bard">
|
||||||
|
<span class="name">Bard</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="machinist">
|
||||||
|
<span class="name">Machinist</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="dancer">
|
||||||
|
<span class="name">Dancer</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="black mage">
|
||||||
|
<span class="name">Black Mage</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="summoner">
|
||||||
|
<span class="name">Summoner</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="red mage">
|
||||||
|
<span class="name">Red Mage</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="pictomancer">
|
||||||
|
<span class="name">Pictomancer</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="blue mage (limited job)">
|
||||||
|
<span class="name">Blue Mage</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div>
|
||||||
|
<span>Disciple of Land</span>
|
||||||
|
<div class="jobs">
|
||||||
|
<div class="job" data-job="miner">
|
||||||
|
<span class="name">Miner</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="botanist">
|
||||||
|
<span class="name">Botanist</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="fisher">
|
||||||
|
<span class="name">Fisher</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span>Disciple of Hand</span>
|
||||||
|
<div class="jobs">
|
||||||
|
<div class="job" data-job="carpenter">
|
||||||
|
<span class="name">Carpenter</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="blacksmith">
|
||||||
|
<span class="name">Blacksmith</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="armorer">
|
||||||
|
<span class="name">Armorer</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="goldsmith">
|
||||||
|
<span class="name">Goldsmith</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="leatherworker">
|
||||||
|
<span class="name">Leatherworker</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="weaver">
|
||||||
|
<span class="name">Weaver</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="alchemist">
|
||||||
|
<span class="name">Alchemist</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
<div class="job" data-job="culinarian">
|
||||||
|
<span class="name">Culinarian</span>
|
||||||
|
<span class="level">??</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</BaseLayout>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
setPlaceholders(loadDataFromLocalStorage())
|
||||||
|
|
||||||
|
const request = await fetch("https://api.youhavetrouble.me/games/ffxiv/").catch(() => null);
|
||||||
|
if (request !== null && request.ok) {
|
||||||
|
const data = await request.json();
|
||||||
|
setPlaceholders(data);
|
||||||
|
saveDataToLocalStorage(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setPlaceholders(data: any) {
|
||||||
|
if (data === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const profilePic = document.querySelector(".character-profile .portrait img") as HTMLImageElement | null;
|
||||||
|
if (profilePic) {
|
||||||
|
profilePic.src = data?.portrait_url;
|
||||||
|
}
|
||||||
|
|
||||||
|
const nameElem = document.querySelector(".character-profile .description .info .name") as HTMLElement | null;
|
||||||
|
if (nameElem) {
|
||||||
|
nameElem.textContent = data?.name ?? "???";
|
||||||
|
}
|
||||||
|
const serverElem = document.querySelector(".character-profile .description .info .technical .server") as HTMLElement | null;
|
||||||
|
if (serverElem) {
|
||||||
|
serverElem.textContent = data?.server ?? "???";
|
||||||
|
}
|
||||||
|
const datacenterElem = document.querySelector(".character-profile .description .info .technical .datacenter") as HTMLElement | null;
|
||||||
|
if (datacenterElem) {
|
||||||
|
datacenterElem.textContent = `(${data?.datacenter ?? "???"})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const jobs = data?.jobs ?? [];
|
||||||
|
if (Array.isArray(jobs)) {
|
||||||
|
jobs.forEach((job: {name: string, level: number}) => {
|
||||||
|
const name = job.name.toLowerCase().split(" / ")[0];
|
||||||
|
const jobElem = document.querySelector(`.character-profile .description .jobs .job[data-job="${name}"]`);
|
||||||
|
if (jobElem) {
|
||||||
|
const levelElem = jobElem.querySelector(".level") as HTMLElement | null;
|
||||||
|
if (levelElem) {
|
||||||
|
levelElem.textContent = job.level.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function saveDataToLocalStorage(data: any) {
|
||||||
|
try {
|
||||||
|
localStorage.setItem('ffxivCharacterData', JSON.stringify(data));
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Error saving FFXIV character data to localStorage", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function loadDataFromLocalStorage() {
|
||||||
|
try {
|
||||||
|
const data = localStorage.getItem('ffxivCharacterData');
|
||||||
|
if (data) {
|
||||||
|
return JSON.parse(data);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Error loading FFXIV character data from localStorage", e);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.character-profile {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-items: flex-start;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 2rem;
|
||||||
|
gap: 2rem;
|
||||||
|
|
||||||
|
.portrait {
|
||||||
|
width: 300px;
|
||||||
|
height: 400px;
|
||||||
|
border: 2px solid var(--primary-color);
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
object-fit: cover;
|
||||||
|
object-position: top;
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.description {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1rem;
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
|
.info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
.name {
|
||||||
|
font-size: 2rem;
|
||||||
|
font-weight: bold;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.technical {
|
||||||
|
font-size: 1rem;
|
||||||
|
color: #9d9d9d;
|
||||||
|
display: flex;
|
||||||
|
gap: 0.5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.jobs {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(12rem, 1fr));
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5rem;
|
||||||
|
|
||||||
|
.job {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0.5rem;
|
||||||
|
border: 2px solid var(--primary-color);
|
||||||
|
border-radius: 4px;
|
||||||
|
|
||||||
|
.name {
|
||||||
|
font-weight: 600;
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@container (max-width: 800px) {
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
.portrait {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.description {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.info {
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
+22
-5
@@ -36,6 +36,21 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
</div>
|
</div>
|
||||||
<div class="window-row">
|
<div class="window-row">
|
||||||
<div class="window socials" data-title="Socials" id="socials" aria-label="Socials">
|
<div class="window socials" data-title="Socials" id="socials" aria-label="Socials">
|
||||||
|
<div class="buttons">
|
||||||
|
<button popovertarget="socials-info" popovertargetaction="show" tabindex="0" title="Socials info">
|
||||||
|
<span class="icon">ℹ️</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div popover="auto" id="socials-info" class="window">
|
||||||
|
<div class="buttons">
|
||||||
|
<button popovertarget="socials-info" popovertargetaction="hide" tabindex="0"
|
||||||
|
aria-label="Close socials info">
|
||||||
|
<span class="icon">❌</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<h2>Socials</h2>
|
||||||
|
<p>I don't use social media much. Many of the ones listed here are here just as a way to reserve my username<br/> and/or to make it easier to verify that the account belongs to me.</p>
|
||||||
|
</div>
|
||||||
<SocialsWidget/>
|
<SocialsWidget/>
|
||||||
</div>
|
</div>
|
||||||
<div class="window blog" aria-label="Blog" id="blog" data-title="Blog">
|
<div class="window blog" aria-label="Blog" id="blog" data-title="Blog">
|
||||||
@@ -49,7 +64,7 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
) : null
|
) : null
|
||||||
}
|
}
|
||||||
<div class="latest-article">
|
<div class="latest-article">
|
||||||
<span class="title">{latestBlogPost?.data.title}</span>
|
<span class="title" style={`view-transition-name: blog-post-${latestBlogPost?.slug}`}>{latestBlogPost?.data.title}</span>
|
||||||
<span class="excerpt">{latestBlogPost?.data.description}</span>
|
<span class="excerpt">{latestBlogPost?.data.description}</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="actions">
|
<div class="actions">
|
||||||
@@ -87,7 +102,7 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
<div class="window-row">
|
<div class="window-row">
|
||||||
<div class="window" data-title="Projects" id="projects" aria-label="Projects">
|
<div class="window" data-title="Projects" id="projects" aria-label="Projects">
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<button popovertarget="projects-info" popovertargetaction="show" tabindex="0" aria-label="Projects info">
|
<button popovertarget="projects-info" popovertargetaction="show" tabindex="0" title="Projects info">
|
||||||
<span class="icon">ℹ️</span>
|
<span class="icon">ℹ️</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -122,7 +137,7 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
|
|
||||||
#projects-info {
|
#projects-info, #socials-info {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -178,7 +193,6 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
.title {
|
.title {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
font-size: 1.1rem;
|
font-size: 1.1rem;
|
||||||
view-transition-name: blog-title;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.excerpt {
|
.excerpt {
|
||||||
@@ -195,13 +209,16 @@ const latestBlogPost = await getCollection("posts")
|
|||||||
}
|
}
|
||||||
|
|
||||||
.buttons {
|
.buttons {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
.rss {
|
.rss {
|
||||||
display: flex;
|
display: flex;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
aspect-ratio: 1/1;
|
aspect-ratio: 1/1;
|
||||||
padding-bottom: 4px;
|
|
||||||
a {
|
a {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
---
|
||||||
|
import {type CollectionEntry, getCollection} from "astro:content";
|
||||||
|
import BaseLayout from "../../layouts/BaseLayout.astro";
|
||||||
|
|
||||||
|
export async function getStaticPaths() {
|
||||||
|
|
||||||
|
const posts = await getCollection('writing');
|
||||||
|
return posts.map(p => ({
|
||||||
|
params: {
|
||||||
|
slug: p.slug
|
||||||
|
},
|
||||||
|
props: {
|
||||||
|
title: p.data.title,
|
||||||
|
slug: p.slug,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
const {slug, title} = Astro.props;
|
||||||
|
const permalink = `${Astro?.site?.href}writing/${slug}`;
|
||||||
|
const writing = await getCollection('writing');
|
||||||
|
const entry: CollectionEntry<'writing'> | undefined = writing.find(e => e.slug === slug);
|
||||||
|
if (!entry) throw new Error(`Entry not found: ${slug}`);
|
||||||
|
const { Content } = await entry.render();
|
||||||
|
---
|
||||||
|
<BaseLayout title={title} description={""} permalink={permalink} current="writing">
|
||||||
|
<header>
|
||||||
|
<h1 style={`view-transition-name: writing-entry-${slug}`}>{title}</h1>
|
||||||
|
<hr/>
|
||||||
|
</header>
|
||||||
|
<div class="container writing">
|
||||||
|
<article class="content">
|
||||||
|
<Content />
|
||||||
|
</article>
|
||||||
|
<hr/>
|
||||||
|
</div>
|
||||||
|
</BaseLayout>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
|
||||||
|
p {
|
||||||
|
background: red;
|
||||||
|
}
|
||||||
|
|
||||||
|
header {
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
margin-bottom: 0.7em;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
width: fit-content;
|
||||||
|
margin-inline: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
min-width: 100px;
|
||||||
|
width: 30%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
---
|
||||||
|
import BaseLayout from '../../layouts/BaseLayout.astro';
|
||||||
|
import {getCollection} from "astro:content";
|
||||||
|
|
||||||
|
const title = 'Writing';
|
||||||
|
const description = 'Something that\'s supposed to be storytelling';
|
||||||
|
const permalink = `${Astro?.site?.href}writing`;
|
||||||
|
|
||||||
|
const posts= await getCollection('writing');
|
||||||
|
const allPosts= posts.sort((a, b) => new Date(b.data.publishDate).valueOf() - new Date(a.data.publishDate).valueOf());
|
||||||
|
|
||||||
|
const categorizedPosts: Map<string, Array<any>> = new Map();
|
||||||
|
allPosts.forEach(post => {
|
||||||
|
const category = post.data.category || 'Uncategorized';
|
||||||
|
if (!categorizedPosts.has(category)) {
|
||||||
|
categorizedPosts.set(category, []);
|
||||||
|
}
|
||||||
|
categorizedPosts.get(category)?.push(post);
|
||||||
|
})
|
||||||
|
---
|
||||||
|
|
||||||
|
<BaseLayout title={title} description={description} permalink={permalink} current="writing">
|
||||||
|
<div class="container">
|
||||||
|
<h1>Writing</h1>
|
||||||
|
{allPosts.length === 0 && <p>No posts as of yet, hop back later!</p>}
|
||||||
|
|
||||||
|
{
|
||||||
|
categorizedPosts.keys().toArray().map(category => {
|
||||||
|
const posts = categorizedPosts.get(category) || [];
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<h2 class="category">{category}</h2>
|
||||||
|
<ul>
|
||||||
|
{posts.map((post, index) => {
|
||||||
|
const href = `/writing/${post.slug}`;
|
||||||
|
return (
|
||||||
|
<li class="post-item">
|
||||||
|
<a data-astro-prefetch href={href} style={`view-transition-name: writing-entry-${post.slug}`}>{post.data.title}</a>
|
||||||
|
</li>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</BaseLayout>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
h2,
|
||||||
|
.post-item-footer {
|
||||||
|
font-family: var(--font-family-sans);
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
text-transform: capitalize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.post-item-date {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
text-align: left;
|
||||||
|
text-transform: uppercase;
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.post-item {
|
||||||
|
content-visibility: auto;
|
||||||
|
contain-intrinsic-size: 228px;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
margin: 60px auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
padding-left: 0;
|
||||||
|
li {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
+12
-7
@@ -121,10 +121,16 @@ p,
|
|||||||
ul,
|
ul,
|
||||||
ol {
|
ol {
|
||||||
font-size: 1.3rem;
|
font-size: 1.3rem;
|
||||||
line-height: 1.75em;
|
line-height: 2.25rem;
|
||||||
margin: 1.2em 0;
|
margin: 1.2em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.writing {
|
||||||
|
p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ol,
|
ol,
|
||||||
ul {
|
ul {
|
||||||
padding-left: 2rem;
|
padding-left: 2rem;
|
||||||
@@ -219,7 +225,6 @@ figure {
|
|||||||
padding: 0;
|
padding: 0;
|
||||||
border: 0;
|
border: 0;
|
||||||
font-size: 100%;
|
font-size: 100%;
|
||||||
font: inherit;
|
|
||||||
vertical-align: baseline;
|
vertical-align: baseline;
|
||||||
-webkit-margin-start: 0;
|
-webkit-margin-start: 0;
|
||||||
-webkit-margin-end: 0;
|
-webkit-margin-end: 0;
|
||||||
@@ -285,7 +290,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
[popover]::backdrop {
|
[popover]::backdrop {
|
||||||
background-color: rgba(0, 0, 0, 0.5);
|
background-color: rgba(49, 49, 49, 0.5);
|
||||||
backdrop-filter: blur(2px);
|
backdrop-filter: blur(2px);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -314,13 +319,13 @@ button {
|
|||||||
|
|
||||||
.window {
|
.window {
|
||||||
border: 2px solid #d0d0d0;
|
border: 2px solid #d0d0d0;
|
||||||
border-radius: 0.5rem;
|
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
position: relative;
|
position: relative;
|
||||||
overflow-y: hidden;
|
overflow-y: hidden;
|
||||||
padding-top: 1.5rem;
|
padding-top: 1.5rem;
|
||||||
min-width: 10rem;
|
min-width: 10rem;
|
||||||
background-color: #232222;
|
background-color: rgba(17, 17, 17, 0.95);
|
||||||
|
backdrop-filter: blur(4px);
|
||||||
color: #f3f3f3;
|
color: #f3f3f3;
|
||||||
container-type: normal;
|
container-type: normal;
|
||||||
&::before {
|
&::before {
|
||||||
@@ -328,7 +333,7 @@ button {
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
height: 1.5rem;
|
height: 1.75rem;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border-bottom: #d0d0d0 solid 2px;
|
border-bottom: #d0d0d0 solid 2px;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -351,6 +356,6 @@ button {
|
|||||||
top: 0;
|
top: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
font-size: 0.75rem;
|
font-size: 0.75rem;
|
||||||
height: 1.5rem;
|
height: 1.6rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user