This commit is contained in:
		@@ -3,54 +3,100 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<Layout title="404 - Page Not Found">
 | 
			
		||||
  <div class="relative flex flex-col items-center justify-center min-h-[80vh] py-20 text-center px-4 overflow-hidden">
 | 
			
		||||
  <div
 | 
			
		||||
    class="relative flex min-h-[80vh] flex-col items-center justify-center overflow-hidden px-4 py-20 text-center"
 | 
			
		||||
  >
 | 
			
		||||
    <!-- Animated background elements -->
 | 
			
		||||
    <div class="absolute inset-0 overflow-hidden">
 | 
			
		||||
      <div class="absolute -top-20 -left-20 w-64 h-64 bg-zinc-100 dark:bg-zinc-800/50 rounded-full blur-3xl opacity-50 animate-blob"></div>
 | 
			
		||||
      <div class="absolute top-1/2 right-1/4 w-96 h-96 bg-zinc-200 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob animation-delay-2000"></div>
 | 
			
		||||
      <div class="absolute bottom-20 left-1/3 w-72 h-72 bg-zinc-100 dark:bg-zinc-800/40 rounded-full blur-3xl opacity-40 animate-blob animation-delay-4000"></div>
 | 
			
		||||
    </div>
 | 
			
		||||
    
 | 
			
		||||
    <!-- Main content with animation -->
 | 
			
		||||
    <div class="relative z-10 max-w-xl mx-auto">
 | 
			
		||||
      <div class="glitch-wrapper">
 | 
			
		||||
        <h1 class="glitch text-9xl sm:text-[12rem] font-bold text-zinc-900 dark:text-zinc-100 leading-none" data-text="404">404</h1>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob absolute -left-20 -top-20 h-64 w-64 rounded-full bg-zinc-100 opacity-50 blur-3xl dark:bg-zinc-800/50"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      
 | 
			
		||||
      <h2 class="mt-6 text-2xl sm:text-3xl font-bold text-zinc-800 dark:text-zinc-200">Page Not Found</h2>
 | 
			
		||||
      
 | 
			
		||||
      <p class="mt-6 text-zinc-600 dark:text-zinc-400 max-w-md mx-auto text-lg">
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 absolute right-1/4 top-1/2 h-96 w-96 rounded-full bg-zinc-200 opacity-30 blur-3xl dark:bg-zinc-800/30"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-4000 absolute bottom-20 left-1/3 h-72 w-72 rounded-full bg-zinc-100 opacity-40 blur-3xl dark:bg-zinc-800/40"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Main content with animation -->
 | 
			
		||||
    <div class="relative z-10 mx-auto max-w-xl">
 | 
			
		||||
      <div class="glitch-wrapper">
 | 
			
		||||
        <h1
 | 
			
		||||
          class="glitch text-9xl font-bold leading-none text-zinc-900 dark:text-zinc-100 sm:text-[12rem]"
 | 
			
		||||
          data-text="404"
 | 
			
		||||
        >
 | 
			
		||||
          404
 | 
			
		||||
        </h1>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <h2 class="mt-6 text-2xl font-bold text-zinc-800 dark:text-zinc-200 sm:text-3xl">
 | 
			
		||||
        Page Not Found
 | 
			
		||||
      </h2>
 | 
			
		||||
 | 
			
		||||
      <p class="mx-auto mt-6 max-w-md text-lg text-zinc-600 dark:text-zinc-400">
 | 
			
		||||
        The page you're looking for does not exist.
 | 
			
		||||
      </p>
 | 
			
		||||
      
 | 
			
		||||
      <div class="mt-10 flex flex-col sm:flex-row items-center justify-center gap-4">
 | 
			
		||||
 | 
			
		||||
      <div class="mt-10 flex flex-col items-center justify-center gap-4 sm:flex-row">
 | 
			
		||||
        <a
 | 
			
		||||
          href="/"
 | 
			
		||||
          class="group relative inline-flex items-center gap-2 px-6 py-3 rounded-lg bg-zinc-900 text-zinc-100 hover:bg-zinc-800 dark:bg-zinc-100 dark:text-zinc-900 dark:hover:bg-zinc-200 transition-all duration-300 overflow-hidden shadow-lg hover:shadow-xl"
 | 
			
		||||
          class="group relative inline-flex items-center gap-2 overflow-hidden rounded-lg bg-zinc-900 px-6 py-3 text-zinc-100 shadow-lg transition-all duration-300 hover:bg-zinc-800 hover:shadow-xl dark:bg-zinc-100 dark:text-zinc-900 dark:hover:bg-zinc-200"
 | 
			
		||||
        >
 | 
			
		||||
          <span class="absolute inset-0 bg-gradient-to-r from-zinc-700 to-zinc-900 dark:from-zinc-300 dark:to-zinc-100 opacity-0 group-hover:opacity-100 transition-opacity duration-300 z-0"></span>
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" class="w-5 h-5 relative z-10">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M2.25 12l8.954-8.955c.44-.439 1.152-.439 1.591 0L21.75 12M4.5 9.75v10.125c0 .621.504 1.125 1.125 1.125H9.75v-4.875c0-.621.504-1.125 1.125-1.125h2.25c.621 0 1.125.504 1.125 1.125V21h4.125c.621 0 1.125-.504 1.125-1.125V9.75M8.25 21h8.25" />
 | 
			
		||||
          <span
 | 
			
		||||
            class="absolute inset-0 z-0 bg-gradient-to-r from-zinc-700 to-zinc-900 opacity-0 transition-opacity duration-300 group-hover:opacity-100 dark:from-zinc-300 dark:to-zinc-100"
 | 
			
		||||
          ></span>
 | 
			
		||||
          <svg
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
            fill="none"
 | 
			
		||||
            viewBox="0 0 24 24"
 | 
			
		||||
            stroke-width="2"
 | 
			
		||||
            stroke="currentColor"
 | 
			
		||||
            class="relative z-10 h-5 w-5"
 | 
			
		||||
          >
 | 
			
		||||
            <path
 | 
			
		||||
              stroke-linecap="round"
 | 
			
		||||
              stroke-linejoin="round"
 | 
			
		||||
              d="M2.25 12l8.954-8.955c.44-.439 1.152-.439 1.591 0L21.75 12M4.5 9.75v10.125c0 .621.504 1.125 1.125 1.125H9.75v-4.875c0-.621.504-1.125 1.125-1.125h2.25c.621 0 1.125.504 1.125 1.125V21h4.125c.621 0 1.125-.504 1.125-1.125V9.75M8.25 21h8.25"
 | 
			
		||||
            ></path>
 | 
			
		||||
          </svg>
 | 
			
		||||
          <span class="font-medium relative z-10">Return Home</span>
 | 
			
		||||
          <span class="relative z-10 font-medium">Return Home</span>
 | 
			
		||||
        </a>
 | 
			
		||||
        
 | 
			
		||||
        <button 
 | 
			
		||||
 | 
			
		||||
        <button
 | 
			
		||||
          id="back-button"
 | 
			
		||||
          class="group inline-flex items-center gap-2 px-6 py-3 rounded-lg border border-zinc-300 dark:border-zinc-700 text-zinc-700 dark:text-zinc-300 hover:bg-zinc-100 dark:hover:bg-zinc-800 transition-all duration-300 shadow-sm hover:shadow-md"
 | 
			
		||||
          class="group inline-flex items-center gap-2 rounded-lg border border-zinc-300 px-6 py-3 text-zinc-700 shadow-sm transition-all duration-300 hover:bg-zinc-100 hover:shadow-md dark:border-zinc-700 dark:text-zinc-300 dark:hover:bg-zinc-800"
 | 
			
		||||
        >
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" class="w-5 h-5 transition-transform duration-300 group-hover:-translate-x-1">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M10.5 19.5L3 12m0 0l7.5-7.5M3 12h18" />
 | 
			
		||||
          <svg
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
            fill="none"
 | 
			
		||||
            viewBox="0 0 24 24"
 | 
			
		||||
            stroke-width="2"
 | 
			
		||||
            stroke="currentColor"
 | 
			
		||||
            class="h-5 w-5 transition-transform duration-300 group-hover:-translate-x-1"
 | 
			
		||||
          >
 | 
			
		||||
            <path
 | 
			
		||||
              stroke-linecap="round"
 | 
			
		||||
              stroke-linejoin="round"
 | 
			
		||||
              d="M10.5 19.5L3 12m0 0l7.5-7.5M3 12h18"></path>
 | 
			
		||||
          </svg>
 | 
			
		||||
          <span class="font-medium">Go Back</span>
 | 
			
		||||
        </button>
 | 
			
		||||
      </div>
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      <!-- Random fun fact -->
 | 
			
		||||
      <div class="mt-16 p-6 bg-zinc-50 dark:bg-zinc-800/50 rounded-xl shadow-sm max-w-md mx-auto backdrop-blur-sm border border-zinc-100 dark:border-zinc-700/50">
 | 
			
		||||
        <h3 class="text-sm font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Did you know?</h3>
 | 
			
		||||
        <p class="mt-2 text-zinc-700 dark:text-zinc-300 text-sm" id="fun-fact">
 | 
			
		||||
          The 404 error code originated when CERN's web server displayed room 404 (their server room) as the error message when a file wasn't found.
 | 
			
		||||
      <div
 | 
			
		||||
        class="mx-auto mt-16 max-w-md rounded-xl border border-zinc-100 bg-zinc-50 p-6 shadow-sm backdrop-blur-sm dark:border-zinc-700/50 dark:bg-zinc-800/50"
 | 
			
		||||
      >
 | 
			
		||||
        <h3 class="text-sm font-medium uppercase tracking-wider text-zinc-500 dark:text-zinc-400">
 | 
			
		||||
          Did you know?
 | 
			
		||||
        </h3>
 | 
			
		||||
        <p class="mt-2 text-sm text-zinc-700 dark:text-zinc-300" id="fun-fact">
 | 
			
		||||
          The 404 error code originated when CERN's web server displayed room 404 (their server
 | 
			
		||||
          room) as the error message when a file wasn't found.
 | 
			
		||||
        </p>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
@@ -62,55 +108,57 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
  document.getElementById('back-button')?.addEventListener('click', () => {
 | 
			
		||||
    window.history.back();
 | 
			
		||||
  });
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Array of fun 404 facts
 | 
			
		||||
  const funFacts = [
 | 
			
		||||
    "The 404 error code originated when CERN's web server displayed room 404 (their server room) as the error message when a file wasn't found.",
 | 
			
		||||
    "In internet slang, '404' has become shorthand for something that's missing or someone who's clueless.",
 | 
			
		||||
    "Some websites turn their 404 pages into games, like Google's Pac-Man 404 page that once existed.",
 | 
			
		||||
    "The first web server was a NeXT computer used by Tim Berners-Lee at CERN, where the 404 error was born.",
 | 
			
		||||
    "Many companies use creative 404 pages as a way to showcase their brand personality and humor.",
 | 
			
		||||
    'The first web server was a NeXT computer used by Tim Berners-Lee at CERN, where the 404 error was born.',
 | 
			
		||||
    'Many companies use creative 404 pages as a way to showcase their brand personality and humor.',
 | 
			
		||||
    "The HTTP 1.0 specification from 1996 officially defined the 404 error as 'Not Found'.",
 | 
			
		||||
    "Studies show that well-designed 404 pages can reduce bounce rates by up to 30%.",
 | 
			
		||||
    "The most common cause of 404 errors is mistyped URLs."
 | 
			
		||||
    'Studies show that well-designed 404 pages can reduce bounce rates by up to 30%.',
 | 
			
		||||
    'The most common cause of 404 errors is mistyped URLs.',
 | 
			
		||||
  ];
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Display a random fun fact
 | 
			
		||||
  const funFactElement = document.getElementById('fun-fact');
 | 
			
		||||
  if (funFactElement) {
 | 
			
		||||
    const randomFact = funFacts[Math.floor(Math.random() * funFacts.length)];
 | 
			
		||||
    funFactElement.textContent = randomFact;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Handle SPA transitions for 404 page
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all internal links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links, external links, or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') || 
 | 
			
		||||
          link.getAttribute('target') === '_blank' || 
 | 
			
		||||
          link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
      if (
 | 
			
		||||
        link.getAttribute('href').includes('#') ||
 | 
			
		||||
        link.getAttribute('target') === '_blank' ||
 | 
			
		||||
        link.hasAttribute('data-spa-handled')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Mark as handled to avoid duplicate listeners
 | 
			
		||||
      link.setAttribute('data-spa-handled', 'true');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      link.addEventListener('click', (e) => {
 | 
			
		||||
        // Don't handle if modifier keys are pressed (for opening in new tab, etc.)
 | 
			
		||||
        if (e.metaKey || e.ctrlKey || e.shiftKey || e.altKey) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetHref = link.getAttribute('href');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Trigger page transition animation
 | 
			
		||||
        const pageTransition = document.getElementById('page-transition');
 | 
			
		||||
        if (pageTransition) {
 | 
			
		||||
          pageTransition.classList.remove('opacity-0');
 | 
			
		||||
          pageTransition.classList.add('opacity-100');
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Navigate after transition effect
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            window.location.href = targetHref;
 | 
			
		||||
@@ -121,7 +169,7 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Re-initialize back button after SPA navigation
 | 
			
		||||
    const backButton = document.getElementById('back-button');
 | 
			
		||||
    if (backButton) {
 | 
			
		||||
@@ -130,13 +178,13 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Initialize on first load
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Re-initialize when content changes via Astro's view transitions
 | 
			
		||||
  document.addEventListener('astro:page-load', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // For compatibility with custom transition system
 | 
			
		||||
  document.addEventListener('page-transition-complete', setupSPATransitions);
 | 
			
		||||
</script>
 | 
			
		||||
@@ -157,31 +205,31 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
      transform: translate(0px, 0px) scale(1);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animate-blob {
 | 
			
		||||
    animation: blob 7s infinite;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animation-delay-2000 {
 | 
			
		||||
    animation-delay: 2s;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animation-delay-4000 {
 | 
			
		||||
    animation-delay: 4s;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Glitch effect for 404 text */
 | 
			
		||||
  .glitch-wrapper {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    display: inline-block;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .glitch {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    display: inline-block;
 | 
			
		||||
    animation: glitch-skew 1s infinite linear alternate-reverse;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .glitch::before,
 | 
			
		||||
  .glitch::after {
 | 
			
		||||
    content: attr(data-text);
 | 
			
		||||
@@ -191,20 +239,22 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    height: 100%;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .glitch::before {
 | 
			
		||||
    left: 2px;
 | 
			
		||||
    text-shadow: -2px 0 #ff00c1;
 | 
			
		||||
    clip: rect(44px, 450px, 56px, 0);
 | 
			
		||||
    animation: glitch-anim 5s infinite linear alternate-reverse;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .glitch::after {
 | 
			
		||||
    left: -2px;
 | 
			
		||||
    text-shadow: -2px 0 #00fff9, 2px 2px #ff00c1;
 | 
			
		||||
    text-shadow:
 | 
			
		||||
      -2px 0 #00fff9,
 | 
			
		||||
      2px 2px #ff00c1;
 | 
			
		||||
    animation: glitch-anim2 1s infinite linear alternate-reverse;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  @keyframes glitch-anim {
 | 
			
		||||
    0% {
 | 
			
		||||
      clip: rect(31px, 9999px, 94px, 0);
 | 
			
		||||
@@ -291,7 +341,7 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
      transform: skew(0.6deg);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  @keyframes glitch-anim2 {
 | 
			
		||||
    0% {
 | 
			
		||||
      clip: rect(65px, 9999px, 65px, 0);
 | 
			
		||||
@@ -378,7 +428,7 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
      transform: skew(0.74deg);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  @keyframes glitch-skew {
 | 
			
		||||
    0% {
 | 
			
		||||
      transform: skew(-1deg);
 | 
			
		||||
@@ -414,4 +464,4 @@ import Layout from '../layouts/Layout.astro';
 | 
			
		||||
      transform: skew(0deg);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -3,54 +3,73 @@ import BaseLayout from '../layouts/BaseLayout.astro';
 | 
			
		||||
import { FaJs, FaReact, FaNodeJs, FaPython } from 'react-icons/fa';
 | 
			
		||||
import { SiTypescript, SiAstro } from 'react-icons/si';
 | 
			
		||||
 | 
			
		||||
import directus from "../../lib/directus"
 | 
			
		||||
import { readSingleton, readItems } from "@directus/sdk";
 | 
			
		||||
import directus from '../../lib/directus';
 | 
			
		||||
import { readSingleton, readItems } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
const global = await directus.request(readSingleton("global"));
 | 
			
		||||
const about = await directus.request(readSingleton("about"));
 | 
			
		||||
const global = await directus.request(readSingleton('global'));
 | 
			
		||||
const about = await directus.request(readSingleton('about'));
 | 
			
		||||
 | 
			
		||||
const skills = await directus.request(
 | 
			
		||||
  readItems("skills", {
 | 
			
		||||
    fields: ['*']
 | 
			
		||||
  readItems('skills', {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
  })
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<BaseLayout title="About Me" description={global.description}>
 | 
			
		||||
  <div class="max-w-6xl mx-auto px-4 sm:px-6 py-8 sm:py-12 md:py-16 theme-transition-all">
 | 
			
		||||
  <div class="theme-transition-all mx-auto max-w-6xl px-4 py-8 sm:px-6 sm:py-12 md:py-16">
 | 
			
		||||
    <!-- Hero Section -->
 | 
			
		||||
    <div class="relative mb-12 sm:mb-16 md:mb-20">
 | 
			
		||||
      <!-- Decorative elements -->
 | 
			
		||||
      <div class="absolute -top-10 sm:-top-20 -left-10 sm:-left-20 w-36 sm:w-48 md:w-72 h-36 sm:h-48 md:h-72 bg-zinc-100 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob theme-transition-bg"></div>
 | 
			
		||||
      <div class="absolute -bottom-10 sm:-bottom-20 -right-10 sm:-right-20 w-36 sm:w-48 md:w-72 h-36 sm:h-48 md:h-72 bg-zinc-200 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob animation-delay-2000 theme-transition-bg"></div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob theme-transition-bg absolute -left-10 -top-10 h-36 w-36 rounded-full bg-zinc-100 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:-left-20 sm:-top-20 sm:h-48 sm:w-48 md:h-72 md:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 theme-transition-bg absolute -bottom-10 -right-10 h-36 w-36 rounded-full bg-zinc-200 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:-bottom-20 sm:-right-20 sm:h-48 sm:w-48 md:h-72 md:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="relative grid grid-cols-1 md:grid-cols-2 gap-8 md:gap-12 items-center">
 | 
			
		||||
        <div class="order-2 md:order-1 text-center md:text-left">
 | 
			
		||||
          <h1 class="text-3xl sm:text-4xl md:text-5xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 mb-4 sm:mb-6 theme-transition-color">
 | 
			
		||||
            Hello, I'm <span class="text-transparent bg-clip-text bg-gradient-to-r from-zinc-500 to-zinc-900 dark:from-zinc-300 dark:to-zinc-100 theme-transition-all">{global.name}</span>
 | 
			
		||||
      <div class="relative grid grid-cols-1 items-center gap-8 md:grid-cols-2 md:gap-12">
 | 
			
		||||
        <div class="order-2 text-center md:order-1 md:text-left">
 | 
			
		||||
          <h1
 | 
			
		||||
            class="theme-transition-color mb-4 text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:mb-6 sm:text-4xl md:text-5xl"
 | 
			
		||||
          >
 | 
			
		||||
            Hello, I'm <span
 | 
			
		||||
              class="theme-transition-all bg-gradient-to-r from-zinc-500 to-zinc-900 bg-clip-text text-transparent dark:from-zinc-300 dark:to-zinc-100"
 | 
			
		||||
              >{global.name}</span
 | 
			
		||||
            >
 | 
			
		||||
          </h1>
 | 
			
		||||
 | 
			
		||||
          <p class="text-lg sm:text-xl text-zinc-600 dark:text-zinc-400 mb-6 sm:mb-8 leading-relaxed theme-transition-color">
 | 
			
		||||
          <p
 | 
			
		||||
            class="theme-transition-color mb-6 text-lg leading-relaxed text-zinc-600 dark:text-zinc-400 sm:mb-8 sm:text-xl"
 | 
			
		||||
          >
 | 
			
		||||
            {about.background}
 | 
			
		||||
          </p>
 | 
			
		||||
 | 
			
		||||
          <div class="flex flex-wrap gap-4 social-links-container justify-center md:justify-start theme-transition-children">
 | 
			
		||||
          <div
 | 
			
		||||
            class="social-links-container theme-transition-children flex flex-wrap justify-center gap-4 md:justify-start"
 | 
			
		||||
          >
 | 
			
		||||
            <!-- Social links remain the same -->
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div class="order-1 md:order-2 relative">
 | 
			
		||||
          <div class="aspect-square w-full max-w-[280px] sm:max-w-[320px] md:max-w-md mx-auto overflow-hidden rounded-3xl border-4 sm:border-8 border-white dark:border-zinc-800 shadow-xl sm:shadow-2xl theme-transition-all">
 | 
			
		||||
        <div class="relative order-1 md:order-2">
 | 
			
		||||
          <div
 | 
			
		||||
            class="theme-transition-all mx-auto aspect-square w-full max-w-[280px] overflow-hidden rounded-3xl border-4 border-white shadow-xl dark:border-zinc-800 sm:max-w-[320px] sm:border-8 sm:shadow-2xl md:max-w-md"
 | 
			
		||||
          >
 | 
			
		||||
            <img
 | 
			
		||||
              src=`${process.env.DIRECTUS_URL ?? "https://directus.alexlebens.dev"}/assets/${global.portrait}`
 | 
			
		||||
              alt={global.portrait_alt}
 | 
			
		||||
              class="w-full h-full object-cover"
 | 
			
		||||
              class="h-full w-full object-cover"
 | 
			
		||||
              loading="eager"
 | 
			
		||||
            />
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <!-- Decorative elements -->
 | 
			
		||||
          <div class="absolute -bottom-4 sm:-bottom-6 -right-4 sm:-right-6 w-16 sm:w-20 md:w-24 h-16 sm:h-20 md:h-24 bg-zinc-100 dark:bg-zinc-800 rounded-full border-2 sm:border-4 border-white dark:border-zinc-900 shadow-lg flex items-center justify-center theme-transition-all">
 | 
			
		||||
          <div
 | 
			
		||||
            class="theme-transition-all absolute -bottom-4 -right-4 flex h-16 w-16 items-center justify-center rounded-full border-2 border-white bg-zinc-100 shadow-lg dark:border-zinc-900 dark:bg-zinc-800 sm:-bottom-6 sm:-right-6 sm:h-20 sm:w-20 sm:border-4 md:h-24 md:w-24"
 | 
			
		||||
          >
 | 
			
		||||
            <span class="text-2xl sm:text-3xl">👋</span>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
@@ -58,87 +77,131 @@ const skills = await directus.request(
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- About Section -->
 | 
			
		||||
    <div class="mb-16 sm:mb-20 md:mb-24 theme-transition-all">
 | 
			
		||||
      <div class="max-w-3xl mx-auto">
 | 
			
		||||
        <h2 class="text-2xl sm:text-3xl font-bold text-zinc-900 dark:text-zinc-100 mb-6 sm:mb-8 flex items-center justify-center md:justify-start theme-transition-color">
 | 
			
		||||
          <span class="hidden sm:inline-block w-8 sm:w-12 h-1 bg-zinc-300 dark:bg-zinc-700 mr-4 theme-transition-bg"></span>
 | 
			
		||||
    <div class="theme-transition-all mb-16 sm:mb-20 md:mb-24">
 | 
			
		||||
      <div class="mx-auto max-w-3xl">
 | 
			
		||||
        <h2
 | 
			
		||||
          class="theme-transition-color mb-6 flex items-center justify-center text-2xl font-bold text-zinc-900 dark:text-zinc-100 sm:mb-8 sm:text-3xl md:justify-start"
 | 
			
		||||
        >
 | 
			
		||||
          <span
 | 
			
		||||
            class="theme-transition-bg mr-4 hidden h-1 w-8 bg-zinc-300 dark:bg-zinc-700 sm:inline-block sm:w-12"
 | 
			
		||||
          ></span>
 | 
			
		||||
          About Me
 | 
			
		||||
          <span class="hidden sm:inline-block w-8 sm:w-12 h-1 bg-zinc-300 dark:bg-zinc-700 ml-4 theme-transition-bg"></span>
 | 
			
		||||
          <span
 | 
			
		||||
            class="theme-transition-bg ml-4 hidden h-1 w-8 bg-zinc-300 dark:bg-zinc-700 sm:inline-block sm:w-12"
 | 
			
		||||
          ></span>
 | 
			
		||||
        </h2>
 | 
			
		||||
 | 
			
		||||
        <div class="prose prose-zinc dark:prose-invert max-w-none theme-transition-all">
 | 
			
		||||
          <p class="text-base sm:text-lg leading-relaxed mb-4 sm:mb-6 theme-transition-color">
 | 
			
		||||
        <div class="theme-transition-all prose prose-zinc max-w-none dark:prose-invert">
 | 
			
		||||
          <p class="theme-transition-color mb-4 text-base leading-relaxed sm:mb-6 sm:text-lg">
 | 
			
		||||
            {about.experience}
 | 
			
		||||
          </p>
 | 
			
		||||
 | 
			
		||||
          <p class="text-base sm:text-lg leading-relaxed mb-4 sm:mb-6 theme-transition-color">
 | 
			
		||||
          <p class="theme-transition-color mb-4 text-base leading-relaxed sm:mb-6 sm:text-lg">
 | 
			
		||||
            {about.education}
 | 
			
		||||
          </p>
 | 
			
		||||
 | 
			
		||||
          <p class="text-base sm:text-lg leading-relaxed mb-4 sm:mb-6 theme-transition-color">
 | 
			
		||||
          <p class="theme-transition-color mb-4 text-base leading-relaxed sm:mb-6 sm:text-lg">
 | 
			
		||||
            {about.certifications}
 | 
			
		||||
          </p>
 | 
			
		||||
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Skills Section -->
 | 
			
		||||
    <div class="mb-16 sm:mb-20 md:mb-24 theme-transition-all">
 | 
			
		||||
      <h2 class="text-2xl sm:text-3xl font-bold text-zinc-900 dark:text-zinc-100 mb-8 sm:mb-12 text-center theme-transition-color">Tech Stack</h2>
 | 
			
		||||
    <div class="theme-transition-all mb-16 sm:mb-20 md:mb-24">
 | 
			
		||||
      <h2
 | 
			
		||||
        class="theme-transition-color mb-8 text-center text-2xl font-bold text-zinc-900 dark:text-zinc-100 sm:mb-12 sm:text-3xl"
 | 
			
		||||
      >
 | 
			
		||||
        Tech Stack
 | 
			
		||||
      </h2>
 | 
			
		||||
 | 
			
		||||
      <div class="tech-stack-slider relative overflow-hidden py-4 sm:py-8">
 | 
			
		||||
        <!-- Main slider container -->
 | 
			
		||||
        <div class="slider-track flex animate-slide">
 | 
			
		||||
          { skills.map((skill, index) => (
 | 
			
		||||
            <div key={`${skill.title}-${index}`} class="skill-card min-w-[220px] sm:min-w-[280px] mx-2 sm:mx-4 bg-white dark:bg-zinc-800/50 rounded-xl border border-zinc-200 dark:border-zinc-700 hover:border-zinc-300 dark:hover:border-zinc-600 transition-all duration-300 hover:shadow-xl transform hover:-translate-y-2 hover:scale-105 theme-transition-element">
 | 
			
		||||
              <div class="p-4 sm:p-6">
 | 
			
		||||
                <div class="flex items-center justify-between mb-4 sm:mb-6">
 | 
			
		||||
                  <div class="flex items-center gap-2 sm:gap-4">
 | 
			
		||||
                    <div class="w-8 h-8 sm:w-12 sm:h-12 flex items-center justify-center bg-zinc-100 dark:bg-zinc-800 rounded-lg text-zinc-800 dark:text-zinc-200 transform transition-transform group-hover:rotate-12 theme-transition-bg theme-transition-color">
 | 
			
		||||
                      <skill.icon size={20} className="sm:text-2xl transform transition-all hover:scale-125" />
 | 
			
		||||
        <div class="slider-track animate-slide flex">
 | 
			
		||||
          {
 | 
			
		||||
            skills.map((skill, index) => (
 | 
			
		||||
              <div
 | 
			
		||||
                key={`${skill.title}-${index}`}
 | 
			
		||||
                class="skill-card theme-transition-element mx-2 min-w-[220px] transform rounded-xl border border-zinc-200 bg-white transition-all duration-300 hover:-translate-y-2 hover:scale-105 hover:border-zinc-300 hover:shadow-xl dark:border-zinc-700 dark:bg-zinc-800/50 dark:hover:border-zinc-600 sm:mx-4 sm:min-w-[280px]"
 | 
			
		||||
              >
 | 
			
		||||
                <div class="p-4 sm:p-6">
 | 
			
		||||
                  <div class="mb-4 flex items-center justify-between sm:mb-6">
 | 
			
		||||
                    <div class="flex items-center gap-2 sm:gap-4">
 | 
			
		||||
                      <div class="theme-transition-bg theme-transition-color flex h-8 w-8 transform items-center justify-center rounded-lg bg-zinc-100 text-zinc-800 transition-transform group-hover:rotate-12 dark:bg-zinc-800 dark:text-zinc-200 sm:h-12 sm:w-12">
 | 
			
		||||
                        <skill.icon
 | 
			
		||||
                          size={20}
 | 
			
		||||
                          className="sm:text-2xl transform transition-all hover:scale-125"
 | 
			
		||||
                        />
 | 
			
		||||
                      </div>
 | 
			
		||||
                      <h3 class="theme-transition-color text-base font-semibold text-zinc-900 dark:text-zinc-100 sm:text-xl">
 | 
			
		||||
                        {skill.title}
 | 
			
		||||
                      </h3>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <h3 class="text-base sm:text-xl font-semibold text-zinc-900 dark:text-zinc-100 theme-transition-color">{skill.title}</h3>
 | 
			
		||||
                    <span class="theme-transition-all rounded-full bg-zinc-100 px-2 py-0.5 font-mono text-xs text-zinc-600 dark:bg-zinc-800 dark:text-zinc-400 sm:px-2.5 sm:py-1 sm:text-sm">
 | 
			
		||||
                      {skill.level}%
 | 
			
		||||
                    </span>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <span class="text-xs sm:text-sm font-mono bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400 px-2 py-0.5 sm:px-2.5 sm:py-1 rounded-full theme-transition-all">{skill.level}%</span>
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="relative h-1.5 sm:h-2 w-full bg-zinc-100 dark:bg-zinc-700 overflow-hidden rounded-full theme-transition-bg">
 | 
			
		||||
                  <div
 | 
			
		||||
                    class="absolute top-0 left-0 h-full bg-gradient-to-r from-zinc-700 via-zinc-600 to-zinc-800 dark:from-zinc-300 dark:via-zinc-400 dark:to-zinc-200 rounded-full transition-all duration-1000 progress-bar-animate theme-transition-bg"
 | 
			
		||||
                    style={`width: ${skill.level}%`}
 | 
			
		||||
                  ></div>
 | 
			
		||||
                </div>
 | 
			
		||||
                  <div class="theme-transition-bg relative h-1.5 w-full overflow-hidden rounded-full bg-zinc-100 dark:bg-zinc-700 sm:h-2">
 | 
			
		||||
                    <div
 | 
			
		||||
                      class="progress-bar-animate theme-transition-bg absolute left-0 top-0 h-full rounded-full bg-gradient-to-r from-zinc-700 via-zinc-600 to-zinc-800 transition-all duration-1000 dark:from-zinc-300 dark:via-zinc-400 dark:to-zinc-200"
 | 
			
		||||
                      style={`width: ${skill.level}%`}
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                <div class="flex justify-between mt-1 sm:mt-2 text-[10px] sm:text-xs text-zinc-400 dark:text-zinc-500 font-mono theme-transition-color">
 | 
			
		||||
                  <span>Beginner</span>
 | 
			
		||||
                  <span>Advanced</span>
 | 
			
		||||
                  <div class="theme-transition-color mt-1 flex justify-between font-mono text-[10px] text-zinc-400 dark:text-zinc-500 sm:mt-2 sm:text-xs">
 | 
			
		||||
                    <span>Beginner</span>
 | 
			
		||||
                    <span>Advanced</span>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          ))}
 | 
			
		||||
            ))
 | 
			
		||||
          }
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <!-- Gradient overlays for smooth fade effect -->
 | 
			
		||||
        <div class="absolute top-0 bottom-0 left-0 w-12 sm:w-24 bg-gradient-to-r from-white dark:from-zinc-900 to-transparent z-10 theme-transition-bg"></div>
 | 
			
		||||
        <div class="absolute top-0 bottom-0 right-0 w-12 sm:w-24 bg-gradient-to-l from-white dark:from-zinc-900 to-transparent z-10 theme-transition-bg"></div>
 | 
			
		||||
        <div
 | 
			
		||||
          class="theme-transition-bg absolute bottom-0 left-0 top-0 z-10 w-12 bg-gradient-to-r from-white to-transparent dark:from-zinc-900 sm:w-24"
 | 
			
		||||
        >
 | 
			
		||||
        </div>
 | 
			
		||||
        <div
 | 
			
		||||
          class="theme-transition-bg absolute bottom-0 right-0 top-0 z-10 w-12 bg-gradient-to-l from-white to-transparent dark:from-zinc-900 sm:w-24"
 | 
			
		||||
        >
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Contact Section -->
 | 
			
		||||
    <div class="max-w-3xl mx-auto text-center theme-transition-all">
 | 
			
		||||
      <h2 class="text-2xl sm:text-3xl font-bold text-zinc-900 dark:text-zinc-100 mb-4 sm:mb-6 theme-transition-color">Get in Touch</h2>
 | 
			
		||||
      <p class="text-base sm:text-lg text-zinc-600 dark:text-zinc-400 mb-6 sm:mb-8 theme-transition-color">
 | 
			
		||||
        I'm always open to new opportunities and collaborations. If you'd like to work together or just say hello,
 | 
			
		||||
        feel free to reach out.
 | 
			
		||||
    <div class="theme-transition-all mx-auto max-w-3xl text-center">
 | 
			
		||||
      <h2
 | 
			
		||||
        class="theme-transition-color mb-4 text-2xl font-bold text-zinc-900 dark:text-zinc-100 sm:mb-6 sm:text-3xl"
 | 
			
		||||
      >
 | 
			
		||||
        Get in Touch
 | 
			
		||||
      </h2>
 | 
			
		||||
      <p
 | 
			
		||||
        class="theme-transition-color mb-6 text-base text-zinc-600 dark:text-zinc-400 sm:mb-8 sm:text-lg"
 | 
			
		||||
      >
 | 
			
		||||
        I'm always open to new opportunities and collaborations. If you'd like to work together or
 | 
			
		||||
        just say hello, feel free to reach out.
 | 
			
		||||
      </p>
 | 
			
		||||
 | 
			
		||||
      <a
 | 
			
		||||
        href=`mailto:${global.email}`
 | 
			
		||||
        class="inline-flex items-center justify-center px-6 sm:px-8 py-3 sm:py-4 rounded-lg bg-zinc-900 dark:bg-zinc-100 text-zinc-100 dark:text-zinc-900 hover:bg-zinc-700 dark:hover:bg-zinc-300 transition-colors text-base sm:text-lg font-medium theme-transition-all"
 | 
			
		||||
        class="theme-transition-all inline-flex items-center justify-center rounded-lg bg-zinc-900 px-6 py-3 text-base font-medium text-zinc-100 transition-colors hover:bg-zinc-700 dark:bg-zinc-100 dark:text-zinc-900 dark:hover:bg-zinc-300 sm:px-8 sm:py-4 sm:text-lg"
 | 
			
		||||
      >
 | 
			
		||||
        <svg xmlns="http://www.w3.org/2000/svg" class="h-4 w-4 sm:h-5 sm:w-5 mr-2" fill="none" viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
          <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 8l7.89 5.26a2 2 0 002.22 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z" />
 | 
			
		||||
        <svg
 | 
			
		||||
          xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
          class="mr-2 h-4 w-4 sm:h-5 sm:w-5"
 | 
			
		||||
          fill="none"
 | 
			
		||||
          viewBox="0 0 24 24"
 | 
			
		||||
          stroke="currentColor"
 | 
			
		||||
        >
 | 
			
		||||
          <path
 | 
			
		||||
            stroke-linecap="round"
 | 
			
		||||
            stroke-linejoin="round"
 | 
			
		||||
            stroke-width="2"
 | 
			
		||||
            d="M3 8l7.89 5.26a2 2 0 002.22 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"
 | 
			
		||||
          ></path>
 | 
			
		||||
        </svg>
 | 
			
		||||
        Say Hello
 | 
			
		||||
      </a>
 | 
			
		||||
@@ -157,7 +220,8 @@ const skills = await directus.request(
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @keyframes blob-bounce {
 | 
			
		||||
    0%, 100% {
 | 
			
		||||
    0%,
 | 
			
		||||
    100% {
 | 
			
		||||
      transform: translate(0, 0) scale(1);
 | 
			
		||||
    }
 | 
			
		||||
    25% {
 | 
			
		||||
@@ -218,7 +282,9 @@ const skills = await directus.request(
 | 
			
		||||
  /* Reduce animation complexity on mobile for better performance */
 | 
			
		||||
  @media (max-width: 640px) {
 | 
			
		||||
    .skill-card {
 | 
			
		||||
      transition: transform 0.3s ease, box-shadow 0.3s ease;
 | 
			
		||||
      transition:
 | 
			
		||||
        transform 0.3s ease,
 | 
			
		||||
        box-shadow 0.3s ease;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .skill-card:hover {
 | 
			
		||||
@@ -234,7 +300,11 @@ const skills = await directus.request(
 | 
			
		||||
    left: -10%;
 | 
			
		||||
    width: 120%;
 | 
			
		||||
    height: 120%;
 | 
			
		||||
    background: radial-gradient(circle at center, rgba(255,255,255,0.1) 0%, rgba(255,255,255,0) 70%);
 | 
			
		||||
    background: radial-gradient(
 | 
			
		||||
      circle at center,
 | 
			
		||||
      rgba(255, 255, 255, 0.1) 0%,
 | 
			
		||||
      rgba(255, 255, 255, 0) 70%
 | 
			
		||||
    );
 | 
			
		||||
    opacity: 0;
 | 
			
		||||
    transition: opacity 0.5s ease;
 | 
			
		||||
    pointer-events: none;
 | 
			
		||||
@@ -256,7 +326,7 @@ const skills = await directus.request(
 | 
			
		||||
    left: -100%;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    height: 100%;
 | 
			
		||||
    background: linear-gradient(90deg, transparent, rgba(255,255,255,0.2), transparent);
 | 
			
		||||
    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);
 | 
			
		||||
    animation: progress-shine 2s infinite;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -271,7 +341,8 @@ const skills = await directus.request(
 | 
			
		||||
 | 
			
		||||
  /* Improved touch targets for mobile */
 | 
			
		||||
  @media (max-width: 640px) {
 | 
			
		||||
    a, button {
 | 
			
		||||
    a,
 | 
			
		||||
    button {
 | 
			
		||||
      min-height: 44px;
 | 
			
		||||
      display: flex;
 | 
			
		||||
      align-items: center;
 | 
			
		||||
@@ -290,11 +361,12 @@ const skills = await directus.request(
 | 
			
		||||
 | 
			
		||||
  /* Smooth card transition during theme switch */
 | 
			
		||||
  .skill-card.theme-transition-element {
 | 
			
		||||
    transition: background-color var(--theme-transition),
 | 
			
		||||
                border-color var(--theme-transition),
 | 
			
		||||
                color var(--theme-transition),
 | 
			
		||||
                box-shadow var(--theme-transition),
 | 
			
		||||
                transform 0.5s cubic-bezier(0.175, 0.885, 0.32, 1.275);
 | 
			
		||||
    transition:
 | 
			
		||||
      background-color var(--theme-transition),
 | 
			
		||||
      border-color var(--theme-transition),
 | 
			
		||||
      color var(--theme-transition),
 | 
			
		||||
      box-shadow var(--theme-transition),
 | 
			
		||||
      transform 0.5s cubic-bezier(0.175, 0.885, 0.32, 1.275);
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
 | 
			
		||||
@@ -350,7 +422,7 @@ const skills = await directus.request(
 | 
			
		||||
    const cards = document.querySelectorAll('.skill-card');
 | 
			
		||||
 | 
			
		||||
    if (!isTouchDevice) {
 | 
			
		||||
      cards.forEach(card => {
 | 
			
		||||
      cards.forEach((card) => {
 | 
			
		||||
        card.addEventListener('mousemove', (e) => {
 | 
			
		||||
          const rect = card.getBoundingClientRect();
 | 
			
		||||
          const x = e.clientX - rect.left;
 | 
			
		||||
@@ -380,7 +452,7 @@ const skills = await directus.request(
 | 
			
		||||
      });
 | 
			
		||||
    } else {
 | 
			
		||||
      // Simpler effects for touch devices
 | 
			
		||||
      cards.forEach(card => {
 | 
			
		||||
      cards.forEach((card) => {
 | 
			
		||||
        card.addEventListener('touchstart', () => {
 | 
			
		||||
          card.classList.add('is-touched');
 | 
			
		||||
        });
 | 
			
		||||
@@ -413,11 +485,13 @@ const skills = await directus.request(
 | 
			
		||||
  // Handle SPA transitions for about page
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all internal links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links, external links, or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') ||
 | 
			
		||||
          link.getAttribute('target') === '_blank' ||
 | 
			
		||||
          link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
      if (
 | 
			
		||||
        link.getAttribute('href').includes('#') ||
 | 
			
		||||
        link.getAttribute('target') === '_blank' ||
 | 
			
		||||
        link.hasAttribute('data-spa-handled')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
@@ -455,9 +529,12 @@ const skills = await directus.request(
 | 
			
		||||
      // Animate hero section elements
 | 
			
		||||
      const heroElements = document.querySelectorAll('h1, .order-2 p, .social-links-container');
 | 
			
		||||
      heroElements.forEach((el, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          el.classList.add('animate-reveal');
 | 
			
		||||
        }, 100 + (index * 150));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            el.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          100 + index * 150
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      // Animate profile image
 | 
			
		||||
@@ -471,17 +548,23 @@ const skills = await directus.request(
 | 
			
		||||
      // Animate skill bars with staggered delay
 | 
			
		||||
      const skillBars = document.querySelectorAll('.skill-bar');
 | 
			
		||||
      skillBars.forEach((bar, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          bar.classList.add('animate-skill');
 | 
			
		||||
        }, 500 + (index * 100));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            bar.classList.add('animate-skill');
 | 
			
		||||
          },
 | 
			
		||||
          500 + index * 100
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      // Animate sections with staggered delay
 | 
			
		||||
      const sections = document.querySelectorAll('section');
 | 
			
		||||
      sections.forEach((section, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          section.classList.add('animate-reveal');
 | 
			
		||||
        }, 300 + (index * 200));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            section.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          300 + index * 200
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,97 +1,144 @@
 | 
			
		||||
---
 | 
			
		||||
import BlogPost from '../../layouts/BlogPost.astro';
 | 
			
		||||
 | 
			
		||||
import directus from "../../../lib/directus"
 | 
			
		||||
import { readItems } from "@directus/sdk";
 | 
			
		||||
import directus from '../../../lib/directus';
 | 
			
		||||
import { readItems } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
export async function getStaticPaths() {
 | 
			
		||||
  const posts = await directus.request(readItems("posts", {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
  }));
 | 
			
		||||
  
 | 
			
		||||
  const posts = await directus.request(
 | 
			
		||||
    readItems('posts', {
 | 
			
		||||
      fields: ['*'],
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const sortedEntries = [...posts].sort(
 | 
			
		||||
    (a, b) => b.published_date.valueOf() - a.published_date.valueOf()
 | 
			
		||||
  );
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  return sortedEntries.map((post, index) => {
 | 
			
		||||
    return {
 | 
			
		||||
      params: { slug: post.slug },
 | 
			
		||||
      props: { 
 | 
			
		||||
      props: {
 | 
			
		||||
        post,
 | 
			
		||||
        nextPost: index > 0 ? sortedEntries[index - 1] : null,
 | 
			
		||||
        prevPost: index < sortedEntries.length - 1 ? sortedEntries[index + 1] : null
 | 
			
		||||
        prevPost: index < sortedEntries.length - 1 ? sortedEntries[index + 1] : null,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<BlogPost slug={post.slug} title={post.title} description={post.description} content={post.content} image={post.image} image_alt={post.image_alt} published_date={post.published_date} updated_date={post.updated_date} tags={post.tags}>
 | 
			
		||||
    <!-- Main Content - Enhanced with better typography and spacing -->
 | 
			
		||||
    <div class="prose prose-zinc dark:prose-invert max-w-none prose-headings:scroll-mt-24 prose-headings:font-semibold prose-a:text-zinc-800 dark:prose-a:text-zinc-300 prose-a:font-medium prose-a:underline-offset-4 hover:prose-a:text-zinc-600 dark:hover:prose-a:text-zinc-100 prose-img:rounded-xl sm:prose-base prose-sm">
 | 
			
		||||
      <div set:html={post.content} />
 | 
			
		||||
    </div>
 | 
			
		||||
<BlogPost
 | 
			
		||||
  slug={post.slug}
 | 
			
		||||
  title={post.title}
 | 
			
		||||
  description={post.description}
 | 
			
		||||
  content={post.content}
 | 
			
		||||
  image={post.image}
 | 
			
		||||
  image_alt={post.image_alt}
 | 
			
		||||
  published_date={post.published_date}
 | 
			
		||||
  updated_date={post.updated_date}
 | 
			
		||||
  tags={post.tags}
 | 
			
		||||
>
 | 
			
		||||
  <!-- Main Content - Enhanced with better typography and spacing -->
 | 
			
		||||
  <div
 | 
			
		||||
    class="prose prose-sm prose-zinc max-w-none dark:prose-invert sm:prose-base prose-headings:scroll-mt-24 prose-headings:font-semibold prose-a:font-medium prose-a:text-zinc-800 prose-a:underline-offset-4 hover:prose-a:text-zinc-600 prose-img:rounded-xl dark:prose-a:text-zinc-300 dark:hover:prose-a:text-zinc-100"
 | 
			
		||||
  >
 | 
			
		||||
    <div set:html={post.content} />
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Next/Previous Navigation - Improved responsive design -->
 | 
			
		||||
    <div class="mt-12 sm:mt-16 grid grid-cols-1 md:grid-cols-2 gap-4 sm:gap-6 border-t border-zinc-200 dark:border-zinc-800 pt-8 sm:pt-12">
 | 
			
		||||
      {prevPost && (
 | 
			
		||||
        <a 
 | 
			
		||||
          href={`/blog/${prevPost.slug}`} 
 | 
			
		||||
          class="group relative flex flex-col h-full p-4 sm:p-6 rounded-xl border border-zinc-200 dark:border-zinc-800 hover:bg-zinc-50 dark:hover:bg-zinc-800/50 transition-all duration-300 hover:-translate-y-1 overflow-hidden"
 | 
			
		||||
  <!-- Next/Previous Navigation - Improved responsive design -->
 | 
			
		||||
  <div
 | 
			
		||||
    class="mt-12 grid grid-cols-1 gap-4 border-t border-zinc-200 pt-8 dark:border-zinc-800 sm:mt-16 sm:gap-6 sm:pt-12 md:grid-cols-2"
 | 
			
		||||
  >
 | 
			
		||||
    {
 | 
			
		||||
      prevPost && (
 | 
			
		||||
        <a
 | 
			
		||||
          href={`/blog/${prevPost.slug}`}
 | 
			
		||||
          class="group relative flex h-full flex-col overflow-hidden rounded-xl border border-zinc-200 p-4 transition-all duration-300 hover:-translate-y-1 hover:bg-zinc-50 dark:border-zinc-800 dark:hover:bg-zinc-800/50 sm:p-6"
 | 
			
		||||
        >
 | 
			
		||||
          <div class="absolute inset-0 bg-gradient-to-r from-zinc-100 to-transparent dark:from-zinc-800 dark:to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300"></div>
 | 
			
		||||
          <span class="relative z-10 text-xs sm:text-sm font-medium text-zinc-500 dark:text-zinc-400 flex items-center gap-1 sm:gap-2 mb-1 sm:mb-2">
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="w-3 h-3 sm:w-4 sm:h-4 transition-transform duration-300 group-hover:-translate-x-1">
 | 
			
		||||
              <path d="m15 18-6-6 6-6"></path>
 | 
			
		||||
          <div class="absolute inset-0 bg-gradient-to-r from-zinc-100 to-transparent opacity-0 transition-opacity duration-300 group-hover:opacity-100 dark:from-zinc-800 dark:to-transparent" />
 | 
			
		||||
          <span class="relative z-10 mb-1 flex items-center gap-1 text-xs font-medium text-zinc-500 dark:text-zinc-400 sm:mb-2 sm:gap-2 sm:text-sm">
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              width="20"
 | 
			
		||||
              height="20"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              stroke-width="2"
 | 
			
		||||
              stroke-linecap="round"
 | 
			
		||||
              stroke-linejoin="round"
 | 
			
		||||
              class="h-3 w-3 transition-transform duration-300 group-hover:-translate-x-1 sm:h-4 sm:w-4"
 | 
			
		||||
            >
 | 
			
		||||
              <path d="m15 18-6-6 6-6" />
 | 
			
		||||
            </svg>
 | 
			
		||||
            Previous Article
 | 
			
		||||
          </span>
 | 
			
		||||
          <h3 class="text-base sm:text-lg font-medium text-zinc-900 dark:text-white line-clamp-2 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors">
 | 
			
		||||
          <h3 class="line-clamp-2 text-base font-medium text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-white dark:group-hover:text-zinc-300 sm:text-lg">
 | 
			
		||||
            {prevPost.title}
 | 
			
		||||
          </h3>
 | 
			
		||||
        </a>
 | 
			
		||||
      )}
 | 
			
		||||
      {nextPost && (
 | 
			
		||||
        <a 
 | 
			
		||||
          href={`/blog/${nextPost.slug}`} 
 | 
			
		||||
          class="group relative flex flex-col h-full p-4 sm:p-6 rounded-xl border border-zinc-200 dark:border-zinc-800 hover:bg-zinc-50 dark:hover:bg-zinc-800/50 transition-all duration-300 hover:-translate-y-1 md:text-right overflow-hidden"
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
    {
 | 
			
		||||
      nextPost && (
 | 
			
		||||
        <a
 | 
			
		||||
          href={`/blog/${nextPost.slug}`}
 | 
			
		||||
          class="group relative flex h-full flex-col overflow-hidden rounded-xl border border-zinc-200 p-4 transition-all duration-300 hover:-translate-y-1 hover:bg-zinc-50 dark:border-zinc-800 dark:hover:bg-zinc-800/50 sm:p-6 md:text-right"
 | 
			
		||||
        >
 | 
			
		||||
          <div class="absolute inset-0 bg-gradient-to-l from-zinc-100 to-transparent dark:from-zinc-800 dark:to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-300"></div>
 | 
			
		||||
          <span class="relative z-10 text-xs sm:text-sm font-medium text-zinc-500 dark:text-zinc-400 flex items-center gap-1 sm:gap-2 mb-1 sm:mb-2 md:justify-end">
 | 
			
		||||
          <div class="absolute inset-0 bg-gradient-to-l from-zinc-100 to-transparent opacity-0 transition-opacity duration-300 group-hover:opacity-100 dark:from-zinc-800 dark:to-transparent" />
 | 
			
		||||
          <span class="relative z-10 mb-1 flex items-center gap-1 text-xs font-medium text-zinc-500 dark:text-zinc-400 sm:mb-2 sm:gap-2 sm:text-sm md:justify-end">
 | 
			
		||||
            Next Article
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="w-3 h-3 sm:w-4 sm:h-4 transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
              <path d="m9 18 6-6-6-6"></path>
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              width="20"
 | 
			
		||||
              height="20"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              stroke-width="2"
 | 
			
		||||
              stroke-linecap="round"
 | 
			
		||||
              stroke-linejoin="round"
 | 
			
		||||
              class="h-3 w-3 transition-transform duration-300 group-hover:translate-x-1 sm:h-4 sm:w-4"
 | 
			
		||||
            >
 | 
			
		||||
              <path d="m9 18 6-6-6-6" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </span>
 | 
			
		||||
          <h3 class="text-base sm:text-lg font-medium text-zinc-900 dark:text-white line-clamp-2 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors">
 | 
			
		||||
          <h3 class="line-clamp-2 text-base font-medium text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-white dark:group-hover:text-zinc-300 sm:text-lg">
 | 
			
		||||
            {nextPost.title}
 | 
			
		||||
          </h3>
 | 
			
		||||
        </a>
 | 
			
		||||
      )}
 | 
			
		||||
    </div>
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
  </div>
 | 
			
		||||
</BlogPost>
 | 
			
		||||
 | 
			
		||||
<script>
 | 
			
		||||
  // Removing TOC-related functions
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Add copy buttons to code blocks
 | 
			
		||||
  function initializeCodeCopyButtons() {
 | 
			
		||||
    const codeBlocks = document.querySelectorAll('pre');
 | 
			
		||||
    
 | 
			
		||||
    codeBlocks.forEach(block => {
 | 
			
		||||
 | 
			
		||||
    codeBlocks.forEach((block) => {
 | 
			
		||||
      // Skip if already processed by either method
 | 
			
		||||
      if (block.classList.contains('code-block-processed') || block.classList.contains('enhanced')) {
 | 
			
		||||
      if (
 | 
			
		||||
        block.classList.contains('code-block-processed') ||
 | 
			
		||||
        block.classList.contains('enhanced')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      block.classList.add('code-block-processed');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Create wrapper if not already wrapped
 | 
			
		||||
      let wrapper;
 | 
			
		||||
      if (block.parentNode.classList.contains('relative') && block.parentNode.classList.contains('group')) {
 | 
			
		||||
      if (
 | 
			
		||||
        block.parentNode.classList.contains('relative') &&
 | 
			
		||||
        block.parentNode.classList.contains('group')
 | 
			
		||||
      ) {
 | 
			
		||||
        wrapper = block.parentNode;
 | 
			
		||||
      } else {
 | 
			
		||||
        wrapper = document.createElement('div');
 | 
			
		||||
@@ -99,28 +146,29 @@ const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
        block.parentNode.insertBefore(wrapper, block);
 | 
			
		||||
        wrapper.appendChild(block);
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Add copy button if not already present
 | 
			
		||||
      if (!wrapper.querySelector('.copy-button') && !wrapper.querySelector('.copy-code-button')) {
 | 
			
		||||
        const copyButton = document.createElement('button');
 | 
			
		||||
        copyButton.className = 'copy-button absolute top-2 right-2 p-1.5 rounded-md bg-zinc-700/50 hover:bg-zinc-700 text-zinc-200 opacity-0 group-hover:opacity-100 transition-opacity duration-200';
 | 
			
		||||
        copyButton.className =
 | 
			
		||||
          'copy-button absolute top-2 right-2 p-1.5 rounded-md bg-zinc-700/50 hover:bg-zinc-700 text-zinc-200 opacity-0 group-hover:opacity-100 transition-opacity duration-200';
 | 
			
		||||
        copyButton.innerHTML = `
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M15.75 17.25v3.375c0 .621-.504 1.125-1.125 1.125h-9.75a1.125 1.125 0 01-1.125-1.125V7.875c0-.621.504-1.125 1.125-1.125H6.75a9.06 9.06 0 011.5.124m7.5 10.376h3.375c.621 0 1.125-.504 1.125-1.125V11.25c0-4.46-3.243-8.161-7.5-8.876a9.06 9.06 0 00-1.5-.124H9.375c-.621 0-1.125.504-1.125 1.125v3.5m7.5 10.375H9.375a1.125 1.125 0 01-1.125-1.125v-9.25m12 6.625v-1.875a3.375 3.375 0 00-3.375-3.375h-1.5a1.125 1.125 0 01-1.125-1.125v-1.5a3.375 3.375 0 00-3.375-3.375H9.75" />
 | 
			
		||||
          </svg>
 | 
			
		||||
        `;
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        copyButton.addEventListener('click', () => {
 | 
			
		||||
          const code = block.querySelector('code').innerText;
 | 
			
		||||
          navigator.clipboard.writeText(code);
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Show copied feedback
 | 
			
		||||
          copyButton.innerHTML = `
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M4.5 12.75l6 6 9-13.5" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          `;
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            copyButton.innerHTML = `
 | 
			
		||||
              <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
 | 
			
		||||
@@ -129,39 +177,39 @@ const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
            `;
 | 
			
		||||
          }, 2000);
 | 
			
		||||
        });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        wrapper.appendChild(copyButton);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Handle SPA transitions for blog post navigation
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle prev/next navigation links
 | 
			
		||||
    document.querySelectorAll('a[href^="/blog/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/blog/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') || link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Mark as handled to avoid duplicate listeners
 | 
			
		||||
      link.setAttribute('data-spa-handled', 'true');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      link.addEventListener('click', (e) => {
 | 
			
		||||
        // Don't handle if modifier keys are pressed (for opening in new tab, etc.)
 | 
			
		||||
        if (e.metaKey || e.ctrlKey || e.shiftKey || e.altKey) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetHref = link.getAttribute('href');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Trigger page transition animation
 | 
			
		||||
        const pageTransition = document.getElementById('page-transition');
 | 
			
		||||
        if (pageTransition) {
 | 
			
		||||
          pageTransition.classList.remove('opacity-0');
 | 
			
		||||
          pageTransition.classList.add('opacity-100');
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Navigate after transition effect
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            window.location.href = targetHref;
 | 
			
		||||
@@ -173,13 +221,13 @@ const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Main initialization function
 | 
			
		||||
  function initializeBlogPost() {
 | 
			
		||||
    // Initialize remaining components
 | 
			
		||||
    initializeCodeCopyButtons();
 | 
			
		||||
    setupSPATransitions();
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Scroll to hash if present in URL
 | 
			
		||||
    if (window.location.hash) {
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
@@ -190,100 +238,105 @@ const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
      }, 100);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Initialize on first load
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', initializeBlogPost);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Re-initialize when content changes via Astro's view transitions
 | 
			
		||||
  document.addEventListener('astro:page-load', initializeBlogPost);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // For compatibility with custom transition system
 | 
			
		||||
  document.addEventListener('page-transition-complete', initializeBlogPost);
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
  /* Removing TOC-related styles */
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Language badge styling */
 | 
			
		||||
  .language-badge {
 | 
			
		||||
    font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
 | 
			
		||||
    font-family:
 | 
			
		||||
      ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', 'Courier New',
 | 
			
		||||
      monospace;
 | 
			
		||||
    text-transform: lowercase;
 | 
			
		||||
    letter-spacing: 0.05em;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Extra small screens */
 | 
			
		||||
  @media (min-width: 480px) {
 | 
			
		||||
    .xs\:inline {
 | 
			
		||||
      display: inline;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:hidden {
 | 
			
		||||
      display: none;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Enhanced typography for blog content - Responsive adjustments */
 | 
			
		||||
  .prose {
 | 
			
		||||
    @apply text-zinc-800 dark:text-zinc-200;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
  .prose h1, .prose h2, .prose h3, .prose h4 {
 | 
			
		||||
    @apply text-zinc-900 dark:text-zinc-100 font-semibold;
 | 
			
		||||
 | 
			
		||||
  .prose h1,
 | 
			
		||||
  .prose h2,
 | 
			
		||||
  .prose h3,
 | 
			
		||||
  .prose h4 {
 | 
			
		||||
    @apply font-semibold text-zinc-900 dark:text-zinc-100;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose h1 {
 | 
			
		||||
    @apply text-2xl sm:text-3xl md:text-4xl;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose h2 {
 | 
			
		||||
    @apply text-xl sm:text-2xl mt-8 sm:mt-12 mb-3 sm:mb-4 pb-2 border-b border-zinc-200 dark:border-zinc-800;
 | 
			
		||||
    @apply mb-3 mt-8 border-b border-zinc-200 pb-2 text-xl dark:border-zinc-800 sm:mb-4 sm:mt-12 sm:text-2xl;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose h3 {
 | 
			
		||||
    @apply text-lg sm:text-xl mt-6 sm:mt-8 mb-2 sm:mb-3;
 | 
			
		||||
    @apply mb-2 mt-6 text-lg sm:mb-3 sm:mt-8 sm:text-xl;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose p {
 | 
			
		||||
    @apply leading-relaxed mb-4 sm:mb-6 text-sm sm:text-base;
 | 
			
		||||
    @apply mb-4 text-sm leading-relaxed sm:mb-6 sm:text-base;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose a {
 | 
			
		||||
    @apply text-zinc-800 dark:text-zinc-300 font-medium underline decoration-zinc-400 dark:decoration-zinc-600 underline-offset-2 hover:text-zinc-600 dark:hover:text-zinc-100 hover:decoration-zinc-600 dark:hover:decoration-zinc-400 transition-colors;
 | 
			
		||||
    @apply font-medium text-zinc-800 underline decoration-zinc-400 underline-offset-2 transition-colors hover:text-zinc-600 hover:decoration-zinc-600 dark:text-zinc-300 dark:decoration-zinc-600 dark:hover:text-zinc-100 dark:hover:decoration-zinc-400;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose blockquote {
 | 
			
		||||
    @apply border-l-4 border-zinc-300 dark:border-zinc-700 pl-4 italic text-zinc-700 dark:text-zinc-300 my-4 sm:my-6;
 | 
			
		||||
    @apply my-4 border-l-4 border-zinc-300 pl-4 italic text-zinc-700 dark:border-zinc-700 dark:text-zinc-300 sm:my-6;
 | 
			
		||||
  }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
  .prose code {
 | 
			
		||||
    @apply bg-zinc-100 dark:bg-zinc-800 px-1.5 py-0.5 rounded text-zinc-800 dark:text-zinc-200 text-sm font-medium;
 | 
			
		||||
    @apply rounded bg-zinc-100 px-1.5 py-0.5 text-sm font-medium text-zinc-800 dark:bg-zinc-800 dark:text-zinc-200;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose pre {
 | 
			
		||||
    @apply bg-[#1e293b] dark:bg-[#1e293b] text-zinc-200 p-3 sm:p-4 rounded-lg overflow-x-auto text-xs sm:text-sm my-4 sm:my-6 shadow-md !important;
 | 
			
		||||
    @apply my-4 overflow-x-auto rounded-lg bg-[#1e293b] p-3 text-xs text-zinc-200 shadow-md dark:bg-[#1e293b] sm:my-6 sm:p-4 sm:text-sm !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose pre code {
 | 
			
		||||
    @apply bg-transparent p-0 text-zinc-200 dark:text-zinc-200 !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose img {
 | 
			
		||||
    @apply rounded-lg shadow-md my-6 sm:my-8 mx-auto max-w-full h-auto;
 | 
			
		||||
    @apply mx-auto my-6 h-auto max-w-full rounded-lg shadow-md sm:my-8;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
  .prose ul, .prose ol {
 | 
			
		||||
    @apply my-4 sm:my-6 pl-5 sm:pl-6;
 | 
			
		||||
 | 
			
		||||
  .prose ul,
 | 
			
		||||
  .prose ol {
 | 
			
		||||
    @apply my-4 pl-5 sm:my-6 sm:pl-6;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose li {
 | 
			
		||||
    @apply mb-1 sm:mb-2 text-sm sm:text-base;
 | 
			
		||||
    @apply mb-1 text-sm sm:mb-2 sm:text-base;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .prose hr {
 | 
			
		||||
    @apply my-8 sm:my-10 border-zinc-200 dark:border-zinc-800;
 | 
			
		||||
    @apply my-8 border-zinc-200 dark:border-zinc-800 sm:my-10;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Line clamp for truncating text */
 | 
			
		||||
  .line-clamp-2 {
 | 
			
		||||
    display: -webkit-box;
 | 
			
		||||
@@ -291,4 +344,4 @@ const { post, nextPost, prevPost } = Astro.props;
 | 
			
		||||
    -webkit-box-orient: vertical;
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,19 +1,17 @@
 | 
			
		||||
---
 | 
			
		||||
import BaseLayout from '../../layouts/BaseLayout.astro';
 | 
			
		||||
 | 
			
		||||
import directus from "../../../lib/directus"
 | 
			
		||||
import { readItems } from "@directus/sdk";
 | 
			
		||||
import directus from '../../../lib/directus';
 | 
			
		||||
import { readItems } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
const posts = await directus.request(
 | 
			
		||||
  readItems("posts", {
 | 
			
		||||
  readItems('posts', {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
    sort: ["-published_date"],
 | 
			
		||||
    sort: ['-published_date'],
 | 
			
		||||
  })
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
const sortedPosts = posts.sort(
 | 
			
		||||
  (a, b) => b.published_date.valueOf() - a.published_date.valueOf()
 | 
			
		||||
);
 | 
			
		||||
const sortedPosts = posts.sort((a, b) => b.published_date.valueOf() - a.published_date.valueOf());
 | 
			
		||||
 | 
			
		||||
// Group posts by year for timeline effect
 | 
			
		||||
const postsByYear = sortedPosts.reduce((acc, post) => {
 | 
			
		||||
@@ -29,176 +27,206 @@ const years = Object.keys(postsByYear).sort((a, b) => b - a);
 | 
			
		||||
const totalPosts = sortedPosts.length;
 | 
			
		||||
 | 
			
		||||
// Get unique tags for search suggestions
 | 
			
		||||
const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
 | 
			
		||||
const allTags = [...new Set(sortedPosts.flatMap((post) => post.tags || []))];
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<BaseLayout title="Blog">
 | 
			
		||||
  <div class="w-full max-w-6xl mx-auto px-4 sm:px-6 py-10 sm:py-16">
 | 
			
		||||
  <div class="mx-auto w-full max-w-6xl px-4 py-10 sm:px-6 sm:py-16">
 | 
			
		||||
    <!-- Header with search  -->
 | 
			
		||||
    <div class="relative mb-12 sm:mb-20">
 | 
			
		||||
      <!-- Decorative elements -->
 | 
			
		||||
      <div class="absolute -top-10 sm:-top-20 -left-10 sm:-left-20 w-48 sm:w-72 h-48 sm:h-72 bg-zinc-100 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob"></div>
 | 
			
		||||
      <div class="absolute -bottom-10 sm:-bottom-20 -right-10 sm:-right-20 w-48 sm:w-72 h-48 sm:h-72 bg-zinc-200 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob animation-delay-2000"></div>
 | 
			
		||||
      
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob absolute -left-10 -top-10 h-48 w-48 rounded-full bg-zinc-100 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:-left-20 sm:-top-20 sm:h-72 sm:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 absolute -bottom-10 -right-10 h-48 w-48 rounded-full bg-zinc-200 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:-bottom-20 sm:-right-20 sm:h-72 sm:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="relative text-center">
 | 
			
		||||
        <h1 class="text-3xl sm:text-4xl md:text-5xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 mb-4">
 | 
			
		||||
        <h1
 | 
			
		||||
          class="mb-4 text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:text-4xl md:text-5xl"
 | 
			
		||||
        >
 | 
			
		||||
          Blog
 | 
			
		||||
        </h1>
 | 
			
		||||
        
 | 
			
		||||
        <p class="text-sm sm:text-base text-zinc-600 dark:text-zinc-400 mb-6 sm:mb-10 max-w-2xl mx-auto">
 | 
			
		||||
 | 
			
		||||
        <p
 | 
			
		||||
          class="mx-auto mb-6 max-w-2xl text-sm text-zinc-600 dark:text-zinc-400 sm:mb-10 sm:text-base"
 | 
			
		||||
        >
 | 
			
		||||
          Thoughts, ideas, and explorations on technology and selfhosting.
 | 
			
		||||
        </p>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    <!-- Grid layout for mobile experience -->
 | 
			
		||||
    <div class="grid grid-cols-1 md:grid-cols-12 gap-6 sm:gap-8">
 | 
			
		||||
    <div class="grid grid-cols-1 gap-6 sm:gap-8 md:grid-cols-12">
 | 
			
		||||
      <!-- Featured post (if exists) -->
 | 
			
		||||
      {sortedPosts.length > 0 && (
 | 
			
		||||
        <div class="md:col-span-12 mb-8 sm:mb-12">
 | 
			
		||||
          <article class="group relative overflow-hidden rounded-none border-b border-zinc-200 dark:border-zinc-800 pb-6 sm:pb-8">            
 | 
			
		||||
            <div class="flex flex-col md:flex-row h-full gap-6 sm:gap-8">
 | 
			
		||||
              {sortedPosts[0].image && (
 | 
			
		||||
                <div class="w-full md:w-1/2 h-60 sm:h-80 md:h-96 overflow-hidden mx-auto md:mx-0 max-w-full sm:max-w-md">
 | 
			
		||||
                  <img
 | 
			
		||||
                    src={`${process.env.DIRECTUS_URL ?? "https://directus.alexlebens.dev"}/assets/${sortedPosts[0].image}`}
 | 
			
		||||
                    alt={sortedPosts[0].title}
 | 
			
		||||
                    class="w-full h-full object-cover grayscale hover:grayscale-0 transition-all duration-700 group-hover:scale-105"
 | 
			
		||||
                    loading="eager"
 | 
			
		||||
                  />
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
              
 | 
			
		||||
              <div class="flex-1 flex flex-col justify-center">
 | 
			
		||||
                <div class="flex items-center text-xs sm:text-sm text-zinc-500 dark:text-zinc-400 gap-2 mb-3 justify-center md:justify-start">
 | 
			
		||||
                  <span class="font-medium uppercase tracking-wider">Featured</span>
 | 
			
		||||
                  <span class="h-px w-6 sm:w-8 bg-zinc-300 dark:bg-zinc-700"></span>
 | 
			
		||||
                  {sortedPosts[0].published_date && (
 | 
			
		||||
                    <time datetime={sortedPosts[0].published_date.toLocaleString()}>
 | 
			
		||||
                      {sortedPosts[0].published_date.toLocaleString('en-US', { 
 | 
			
		||||
                        year: 'numeric', 
 | 
			
		||||
                        month: 'long', 
 | 
			
		||||
                        day: 'numeric' 
 | 
			
		||||
                      })}
 | 
			
		||||
                    </time>
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
                
 | 
			
		||||
                <h2 class="text-2xl sm:text-3xl font-bold text-zinc-900 dark:text-zinc-100 mb-3 sm:mb-4 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors text-center md:text-left">
 | 
			
		||||
                  <a href={`/blog/${sortedPosts[0].slug}/`} class="before:absolute before:inset-0">
 | 
			
		||||
                    {sortedPosts[0].title}
 | 
			
		||||
                  </a>
 | 
			
		||||
                </h2>
 | 
			
		||||
                
 | 
			
		||||
                <p class="text-sm sm:text-base text-zinc-600 dark:text-zinc-400 mb-4 sm:mb-6 line-clamp-3 text-center md:text-left">
 | 
			
		||||
                  {sortedPosts[0].description}
 | 
			
		||||
                </p>
 | 
			
		||||
                
 | 
			
		||||
                <!-- Improved mobile layout for featured post metadata -->
 | 
			
		||||
                <div class="flex items-center gap-3 sm:gap-4 justify-center md:justify-start flex-wrap">                  
 | 
			
		||||
                  {sortedPosts[0].tags && (
 | 
			
		||||
                    <div class="flex flex-wrap gap-2 justify-center md:justify-start">
 | 
			
		||||
                      {sortedPosts[0].tags.slice(0, 2).map((tag) => (
 | 
			
		||||
                        <span class="px-2 sm:px-3 py-1 text-xs uppercase tracking-wider border border-zinc-200 dark:border-zinc-800 text-zinc-600 dark:text-zinc-400">
 | 
			
		||||
                          {tag}
 | 
			
		||||
                        </span>
 | 
			
		||||
                      ))}
 | 
			
		||||
                    </div>
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </article>
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
      
 | 
			
		||||
      <!-- Improved sidebar for mobile -->
 | 
			
		||||
      <div class="md:col-span-3 relative">
 | 
			
		||||
        <div class="md:sticky md:top-24 space-y-4 mb-8 md:mb-0">
 | 
			
		||||
          <h3 class="text-lg font-medium text-zinc-900 dark:text-zinc-100 mb-4 uppercase tracking-wider text-center md:text-left">Archive</h3>
 | 
			
		||||
          
 | 
			
		||||
          <!-- Horizontal scrollable archive on mobile, vertical on desktop -->
 | 
			
		||||
          <div class="flex md:flex-col overflow-x-auto md:overflow-visible pb-4 md:pb-0 hide-scrollbar">
 | 
			
		||||
            {years.map((year, index) => (
 | 
			
		||||
              <a 
 | 
			
		||||
                href={`#year-${year}`}
 | 
			
		||||
                class={`flex items-center py-2 md:py-3 px-4 md:px-0 mr-3 md:mr-0 border-b border-zinc-100 dark:border-zinc-800 hover:bg-zinc-50 dark:hover:bg-zinc-900 transition-colors md:w-full whitespace-nowrap md:whitespace-normal rounded-full md:rounded-none ${index === 0 ? 'bg-zinc-50 dark:bg-zinc-800/50' : ''}`}
 | 
			
		||||
              >
 | 
			
		||||
                <span class="text-base md:text-lg font-medium text-zinc-900 dark:text-zinc-100">{year}</span>
 | 
			
		||||
                <span class="ml-2 md:ml-auto text-xs md:text-sm text-zinc-500 dark:text-zinc-400">
 | 
			
		||||
                  {postsByYear[year].length} post{postsByYear[year].length !== 1 ? 's' : ''}
 | 
			
		||||
                </span>
 | 
			
		||||
              </a>
 | 
			
		||||
            ))}
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
      
 | 
			
		||||
      <!-- Improved post grid for mobile -->
 | 
			
		||||
      <div class="md:col-span-9">
 | 
			
		||||
        {years.map((year) => (
 | 
			
		||||
          <div id={`year-${year}`} class="mb-12 sm:mb-20 scroll-mt-16">
 | 
			
		||||
            <h2 class="text-xl sm:text-2xl font-bold text-zinc-900 dark:text-zinc-100 mb-6 sm:mb-8 border-b border-zinc-200 dark:border-zinc-800 pb-3 sm:pb-4 text-center md:text-left">
 | 
			
		||||
              {year}
 | 
			
		||||
            </h2>
 | 
			
		||||
            
 | 
			
		||||
            <div class={`grid grid-cols-1 ${postsByYear[year].length >= 2 ? 'md:grid-cols-2' : 'md:grid-cols-1'} gap-8 sm:gap-12`}>
 | 
			
		||||
              {postsByYear[year].map((post, index) => (
 | 
			
		||||
                <article class="group relative flex flex-col h-full mx-auto md:mx-0 w-full max-w-sm sm:max-w-md">
 | 
			
		||||
                  {post.image && (
 | 
			
		||||
                    <div class="h-48 sm:h-56 overflow-hidden mb-4 rounded-lg">
 | 
			
		||||
                      <img
 | 
			
		||||
                        src={`${process.env.DIRECTUS_URL ?? "https://directus.alexlebens.dev"}/assets/${post.image}`}
 | 
			
		||||
                        alt={post.title}
 | 
			
		||||
                        class="w-full h-full object-cover grayscale hover:grayscale-0 transition-all duration-700 group-hover:scale-105"
 | 
			
		||||
                        loading="lazy"
 | 
			
		||||
                      />
 | 
			
		||||
                    </div>
 | 
			
		||||
                  )}
 | 
			
		||||
                  
 | 
			
		||||
                  <div class="flex-1 flex flex-col">
 | 
			
		||||
                    <div class="flex items-center text-xs sm:text-sm text-zinc-500 dark:text-zinc-400 gap-3 sm:gap-4 mb-2 sm:mb-3 justify-center md:justify-start flex-wrap">
 | 
			
		||||
                      {post.pubDate && (
 | 
			
		||||
                        <time datetime={post.published_date.toLocaleString()} class="flex items-center">
 | 
			
		||||
                          {post.published_date.toLocaleString('en-US', { 
 | 
			
		||||
                            month: 'short', 
 | 
			
		||||
                            day: 'numeric' 
 | 
			
		||||
                          })}
 | 
			
		||||
                        </time>
 | 
			
		||||
                      )}
 | 
			
		||||
                    </div>
 | 
			
		||||
                    
 | 
			
		||||
                    <h3 class="text-lg sm:text-xl font-semibold text-zinc-900 dark:text-zinc-100 mb-2 sm:mb-3 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors text-center md:text-left">
 | 
			
		||||
                      <a href={`/blog/${post.slug}/`} class="before:absolute before:inset-0">
 | 
			
		||||
                        {post.title}
 | 
			
		||||
                      </a>
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    
 | 
			
		||||
                    <p class="text-sm text-zinc-600 dark:text-zinc-400 mb-4 line-clamp-2 flex-grow text-center md:text-left">
 | 
			
		||||
                      {post.description}
 | 
			
		||||
                    </p>
 | 
			
		||||
                    
 | 
			
		||||
                    {post.tags && (
 | 
			
		||||
                      <div class="flex flex-wrap gap-2 mt-auto justify-center md:justify-start">
 | 
			
		||||
                        {post.tags.slice(0, 2).map((tag) => (
 | 
			
		||||
                          <span class="px-2 sm:px-3 py-1 text-xs uppercase tracking-wider border border-zinc-200 dark:border-zinc-800 text-zinc-600 dark:text-zinc-400">
 | 
			
		||||
      {
 | 
			
		||||
        sortedPosts.length > 0 && (
 | 
			
		||||
          <div class="mb-8 sm:mb-12 md:col-span-12">
 | 
			
		||||
            <article class="group relative overflow-hidden rounded-none border-b border-zinc-200 pb-6 dark:border-zinc-800 sm:pb-8">
 | 
			
		||||
              <div class="flex h-full flex-col gap-6 sm:gap-8 md:flex-row">
 | 
			
		||||
                {sortedPosts[0].image && (
 | 
			
		||||
                  <div class="mx-auto h-60 w-full max-w-full overflow-hidden sm:h-80 sm:max-w-md md:mx-0 md:h-96 md:w-1/2">
 | 
			
		||||
                    <img
 | 
			
		||||
                      src={`${process.env.DIRECTUS_URL ?? 'https://directus.alexlebens.dev'}/assets/${sortedPosts[0].image}`}
 | 
			
		||||
                      alt={sortedPosts[0].title}
 | 
			
		||||
                      class="h-full w-full object-cover grayscale transition-all duration-700 hover:grayscale-0 group-hover:scale-105"
 | 
			
		||||
                      loading="eager"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                )}
 | 
			
		||||
 | 
			
		||||
                <div class="flex flex-1 flex-col justify-center">
 | 
			
		||||
                  <div class="mb-3 flex items-center justify-center gap-2 text-xs text-zinc-500 dark:text-zinc-400 sm:text-sm md:justify-start">
 | 
			
		||||
                    <span class="font-medium uppercase tracking-wider">Featured</span>
 | 
			
		||||
                    <span class="h-px w-6 bg-zinc-300 dark:bg-zinc-700 sm:w-8" />
 | 
			
		||||
                    {sortedPosts[0].published_date && (
 | 
			
		||||
                      <time datetime={sortedPosts[0].published_date.toLocaleString()}>
 | 
			
		||||
                        {sortedPosts[0].published_date.toLocaleString('en-US', {
 | 
			
		||||
                          year: 'numeric',
 | 
			
		||||
                          month: 'long',
 | 
			
		||||
                          day: 'numeric',
 | 
			
		||||
                        })}
 | 
			
		||||
                      </time>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                  <h2 class="mb-3 text-center text-2xl font-bold text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-zinc-100 dark:group-hover:text-zinc-300 sm:mb-4 sm:text-3xl md:text-left">
 | 
			
		||||
                    <a
 | 
			
		||||
                      href={`/blog/${sortedPosts[0].slug}/`}
 | 
			
		||||
                      class="before:absolute before:inset-0"
 | 
			
		||||
                    >
 | 
			
		||||
                      {sortedPosts[0].title}
 | 
			
		||||
                    </a>
 | 
			
		||||
                  </h2>
 | 
			
		||||
 | 
			
		||||
                  <p class="mb-4 line-clamp-3 text-center text-sm text-zinc-600 dark:text-zinc-400 sm:mb-6 sm:text-base md:text-left">
 | 
			
		||||
                    {sortedPosts[0].description}
 | 
			
		||||
                  </p>
 | 
			
		||||
 | 
			
		||||
                  <div class="flex flex-wrap items-center justify-center gap-3 sm:gap-4 md:justify-start">
 | 
			
		||||
                    {sortedPosts[0].tags && (
 | 
			
		||||
                      <div class="flex flex-wrap justify-center gap-2 md:justify-start">
 | 
			
		||||
                        {sortedPosts[0].tags.slice(0, 2).map((tag) => (
 | 
			
		||||
                          <span class="border border-zinc-200 px-2 py-1 text-xs uppercase tracking-wider text-zinc-600 dark:border-zinc-800 dark:text-zinc-400 sm:px-3">
 | 
			
		||||
                            {tag}
 | 
			
		||||
                          </span>
 | 
			
		||||
                        ))}
 | 
			
		||||
                        {post.tags.length > 2 && (
 | 
			
		||||
                          <span class="px-2 sm:px-3 py-1 text-xs uppercase tracking-wider border border-zinc-200 dark:border-zinc-800 text-zinc-600 dark:text-zinc-400">
 | 
			
		||||
                            +{post.tags.length - 2}
 | 
			
		||||
                          </span>
 | 
			
		||||
                        )}
 | 
			
		||||
                      </div>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </div>
 | 
			
		||||
                </article>
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </article>
 | 
			
		||||
          </div>
 | 
			
		||||
        ))}
 | 
			
		||||
        )
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      <!-- Improved sidebar for mobile -->
 | 
			
		||||
      <div class="relative md:col-span-3">
 | 
			
		||||
        <div class="mb-8 space-y-4 md:sticky md:top-24 md:mb-0">
 | 
			
		||||
          <h3
 | 
			
		||||
            class="mb-4 text-center text-lg font-medium uppercase tracking-wider text-zinc-900 dark:text-zinc-100 md:text-left"
 | 
			
		||||
          >
 | 
			
		||||
            Archive
 | 
			
		||||
          </h3>
 | 
			
		||||
 | 
			
		||||
          <!-- Horizontal scrollable archive on mobile, vertical on desktop -->
 | 
			
		||||
          <div
 | 
			
		||||
            class="hide-scrollbar flex overflow-x-auto pb-4 md:flex-col md:overflow-visible md:pb-0"
 | 
			
		||||
          >
 | 
			
		||||
            {
 | 
			
		||||
              years.map((year, index) => (
 | 
			
		||||
                <a
 | 
			
		||||
                  href={`#year-${year}`}
 | 
			
		||||
                  class={`mr-3 flex items-center whitespace-nowrap rounded-full border-b border-zinc-100 px-4 py-2 transition-colors hover:bg-zinc-50 dark:border-zinc-800 dark:hover:bg-zinc-900 md:mr-0 md:w-full md:whitespace-normal md:rounded-none md:px-0 md:py-3 ${index === 0 ? 'bg-zinc-50 dark:bg-zinc-800/50' : ''}`}
 | 
			
		||||
                >
 | 
			
		||||
                  <span class="text-base font-medium text-zinc-900 dark:text-zinc-100 md:text-lg">
 | 
			
		||||
                    {year}
 | 
			
		||||
                  </span>
 | 
			
		||||
                  <span class="ml-2 text-xs text-zinc-500 dark:text-zinc-400 md:ml-auto md:text-sm">
 | 
			
		||||
                    {postsByYear[year].length} post{postsByYear[year].length !== 1 ? 's' : ''}
 | 
			
		||||
                  </span>
 | 
			
		||||
                </a>
 | 
			
		||||
              ))
 | 
			
		||||
            }
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <!-- Improved post grid for mobile -->
 | 
			
		||||
      <div class="md:col-span-9">
 | 
			
		||||
        {
 | 
			
		||||
          years.map((year) => (
 | 
			
		||||
            <div id={`year-${year}`} class="mb-12 scroll-mt-16 sm:mb-20">
 | 
			
		||||
              <h2 class="mb-6 border-b border-zinc-200 pb-3 text-center text-xl font-bold text-zinc-900 dark:border-zinc-800 dark:text-zinc-100 sm:mb-8 sm:pb-4 sm:text-2xl md:text-left">
 | 
			
		||||
                {year}
 | 
			
		||||
              </h2>
 | 
			
		||||
 | 
			
		||||
              <div
 | 
			
		||||
                class={`grid grid-cols-1 ${postsByYear[year].length >= 2 ? 'md:grid-cols-2' : 'md:grid-cols-1'} gap-8 sm:gap-12`}
 | 
			
		||||
              >
 | 
			
		||||
                {postsByYear[year].map((post, index) => (
 | 
			
		||||
                  <article class="group relative mx-auto flex h-full w-full max-w-sm flex-col sm:max-w-md md:mx-0">
 | 
			
		||||
                    {post.image && (
 | 
			
		||||
                      <div class="mb-4 h-48 overflow-hidden rounded-lg sm:h-56">
 | 
			
		||||
                        <img
 | 
			
		||||
                          src={`${process.env.DIRECTUS_URL ?? 'https://directus.alexlebens.dev'}/assets/${post.image}`}
 | 
			
		||||
                          alt={post.title}
 | 
			
		||||
                          class="h-full w-full object-cover grayscale transition-all duration-700 hover:grayscale-0 group-hover:scale-105"
 | 
			
		||||
                          loading="lazy"
 | 
			
		||||
                        />
 | 
			
		||||
                      </div>
 | 
			
		||||
                    )}
 | 
			
		||||
 | 
			
		||||
                    <div class="flex flex-1 flex-col">
 | 
			
		||||
                      <div class="mb-2 flex flex-wrap items-center justify-center gap-3 text-xs text-zinc-500 dark:text-zinc-400 sm:mb-3 sm:gap-4 sm:text-sm md:justify-start">
 | 
			
		||||
                        {post.pubDate && (
 | 
			
		||||
                          <time
 | 
			
		||||
                            datetime={post.published_date.toLocaleString()}
 | 
			
		||||
                            class="flex items-center"
 | 
			
		||||
                          >
 | 
			
		||||
                            {post.published_date.toLocaleString('en-US', {
 | 
			
		||||
                              month: 'short',
 | 
			
		||||
                              day: 'numeric',
 | 
			
		||||
                            })}
 | 
			
		||||
                          </time>
 | 
			
		||||
                        )}
 | 
			
		||||
                      </div>
 | 
			
		||||
 | 
			
		||||
                      <h3 class="mb-2 text-center text-lg font-semibold text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-zinc-100 dark:group-hover:text-zinc-300 sm:mb-3 sm:text-xl md:text-left">
 | 
			
		||||
                        <a href={`/blog/${post.slug}/`} class="before:absolute before:inset-0">
 | 
			
		||||
                          {post.title}
 | 
			
		||||
                        </a>
 | 
			
		||||
                      </h3>
 | 
			
		||||
 | 
			
		||||
                      <p class="mb-4 line-clamp-2 flex-grow text-center text-sm text-zinc-600 dark:text-zinc-400 md:text-left">
 | 
			
		||||
                        {post.description}
 | 
			
		||||
                      </p>
 | 
			
		||||
 | 
			
		||||
                      {post.tags && (
 | 
			
		||||
                        <div class="mt-auto flex flex-wrap justify-center gap-2 md:justify-start">
 | 
			
		||||
                          {post.tags.slice(0, 2).map((tag) => (
 | 
			
		||||
                            <span class="border border-zinc-200 px-2 py-1 text-xs uppercase tracking-wider text-zinc-600 dark:border-zinc-800 dark:text-zinc-400 sm:px-3">
 | 
			
		||||
                              {tag}
 | 
			
		||||
                            </span>
 | 
			
		||||
                          ))}
 | 
			
		||||
                          {post.tags.length > 2 && (
 | 
			
		||||
                            <span class="border border-zinc-200 px-2 py-1 text-xs uppercase tracking-wider text-zinc-600 dark:border-zinc-800 dark:text-zinc-400 sm:px-3">
 | 
			
		||||
                              +{post.tags.length - 2}
 | 
			
		||||
                            </span>
 | 
			
		||||
                          )}
 | 
			
		||||
                        </div>
 | 
			
		||||
                      )}
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </article>
 | 
			
		||||
                ))}
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          ))
 | 
			
		||||
        }
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
@@ -209,13 +237,14 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
  .animate-blob {
 | 
			
		||||
    animation: blob-bounce 8s infinite ease;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animation-delay-2000 {
 | 
			
		||||
    animation-delay: 2s;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  @keyframes blob-bounce {
 | 
			
		||||
    0%, 100% {
 | 
			
		||||
    0%,
 | 
			
		||||
    100% {
 | 
			
		||||
      transform: translate(0, 0) scale(1);
 | 
			
		||||
    }
 | 
			
		||||
    25% {
 | 
			
		||||
@@ -228,36 +257,37 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
      transform: translate(-5%, 5%) scale(0.95);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Search container hover effect */
 | 
			
		||||
  .search-container:hover .search-pulse {
 | 
			
		||||
    animation: pulse 2s cubic-bezier(0.4, 0, 0.6, 1) infinite;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  @keyframes pulse {
 | 
			
		||||
    0%, 100% {
 | 
			
		||||
    0%,
 | 
			
		||||
    100% {
 | 
			
		||||
      opacity: 0;
 | 
			
		||||
    }
 | 
			
		||||
    50% {
 | 
			
		||||
      opacity: 1;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Input focus animation */
 | 
			
		||||
  input:focus + div .search-pulse {
 | 
			
		||||
    animation: pulse 1.5s cubic-bezier(0.4, 0, 0.6, 1) infinite;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Hide scrollbar but keep functionality */
 | 
			
		||||
  .hide-scrollbar {
 | 
			
		||||
    -ms-overflow-style: none;
 | 
			
		||||
    scrollbar-width: none;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .hide-scrollbar::-webkit-scrollbar {
 | 
			
		||||
    display: none;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Line clamp for descriptions */
 | 
			
		||||
  .line-clamp-2 {
 | 
			
		||||
    display: -webkit-box;
 | 
			
		||||
@@ -265,17 +295,18 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
    -webkit-box-orient: vertical;
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .line-clamp-3 {
 | 
			
		||||
    display: -webkit-box;
 | 
			
		||||
    -webkit-line-clamp: 3;
 | 
			
		||||
    -webkit-box-orient: vertical;
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Improved touch targets for mobile */
 | 
			
		||||
  @media (max-width: 640px) {
 | 
			
		||||
    a, button {
 | 
			
		||||
    a,
 | 
			
		||||
    button {
 | 
			
		||||
      min-height: 44px;
 | 
			
		||||
      display: flex;
 | 
			
		||||
      align-items: center;
 | 
			
		||||
@@ -287,7 +318,7 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
  // Script không thay đổi - giữ nguyên chức năng
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', () => {
 | 
			
		||||
    const backToTopButton = document.getElementById('back-to-top');
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    if (backToTopButton) {
 | 
			
		||||
      // Show button when scrolled down
 | 
			
		||||
      const toggleBackToTopButton = () => {
 | 
			
		||||
@@ -299,50 +330,50 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
          backToTopButton.classList.add('opacity-0', 'invisible');
 | 
			
		||||
        }
 | 
			
		||||
      };
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Scroll to top when clicked
 | 
			
		||||
      backToTopButton.addEventListener('click', () => {
 | 
			
		||||
        window.scrollTo({
 | 
			
		||||
          top: 0,
 | 
			
		||||
          behavior: 'smooth'
 | 
			
		||||
          behavior: 'smooth',
 | 
			
		||||
        });
 | 
			
		||||
      });
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Check scroll position
 | 
			
		||||
      window.addEventListener('scroll', toggleBackToTopButton);
 | 
			
		||||
      toggleBackToTopButton(); // Initial check
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Add smooth scrolling to year links
 | 
			
		||||
    document.querySelectorAll('a[href^="#year-"]').forEach(anchor => {
 | 
			
		||||
    document.querySelectorAll('a[href^="#year-"]').forEach((anchor) => {
 | 
			
		||||
      anchor.addEventListener('click', function (e) {
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetId = this.getAttribute('href');
 | 
			
		||||
        const targetElement = document.querySelector(targetId);
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        if (targetElement) {
 | 
			
		||||
          window.scrollTo({
 | 
			
		||||
            top: targetElement.offsetTop - 100,
 | 
			
		||||
            behavior: 'smooth'
 | 
			
		||||
            behavior: 'smooth',
 | 
			
		||||
          });
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Update URL hash without jumping
 | 
			
		||||
          history.pushState(null, null, targetId);
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Add touch support for hover effects
 | 
			
		||||
    const isTouchDevice = 'ontouchstart' in window || navigator.maxTouchPoints > 0;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    if (isTouchDevice) {
 | 
			
		||||
      const articles = document.querySelectorAll('article');
 | 
			
		||||
      
 | 
			
		||||
      articles.forEach(article => {
 | 
			
		||||
 | 
			
		||||
      articles.forEach((article) => {
 | 
			
		||||
        article.addEventListener('touchstart', () => {
 | 
			
		||||
          article.classList.add('is-touched');
 | 
			
		||||
        });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        article.addEventListener('touchend', () => {
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            article.classList.remove('is-touched');
 | 
			
		||||
@@ -351,34 +382,34 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // SPA transition handling
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all blog post links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/blog/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/blog/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') || link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Mark as handled to avoid duplicate listeners
 | 
			
		||||
      link.setAttribute('data-spa-handled', 'true');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      link.addEventListener('click', (e) => {
 | 
			
		||||
        // Don't handle if modifier keys are pressed (for opening in new tab, etc.)
 | 
			
		||||
        if (e.metaKey || e.ctrlKey || e.shiftKey || e.altKey) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetHref = link.getAttribute('href');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Trigger page transition animation
 | 
			
		||||
        const pageTransition = document.getElementById('page-transition');
 | 
			
		||||
        if (pageTransition) {
 | 
			
		||||
          pageTransition.classList.remove('opacity-0');
 | 
			
		||||
          pageTransition.classList.add('opacity-100');
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Navigate after transition effect
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            window.location.href = targetHref;
 | 
			
		||||
@@ -389,19 +420,19 @@ const allTags = [...new Set(sortedPosts.flatMap(post => post.tags || []))];
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Handle year anchor links specially
 | 
			
		||||
    document.querySelectorAll('a[href^="#year-"]').forEach(anchor => {
 | 
			
		||||
    document.querySelectorAll('a[href^="#year-"]').forEach((anchor) => {
 | 
			
		||||
      anchor.setAttribute('data-spa-internal', 'true');
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Initialize on first load
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Re-initialize when content changes via Astro's view transitions
 | 
			
		||||
  document.addEventListener('astro:page-load', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // For compatibility with custom transition system
 | 
			
		||||
  document.addEventListener('page-transition-complete', setupSPATransitions);
 | 
			
		||||
</script>
 | 
			
		||||
</script>
 | 
			
		||||
 
 | 
			
		||||
@@ -2,15 +2,15 @@
 | 
			
		||||
import Layout from '../layouts/Layout.astro';
 | 
			
		||||
import FormattedDate from '../components/FormattedDate.astro';
 | 
			
		||||
 | 
			
		||||
import directus from "../../lib/directus"
 | 
			
		||||
import { readItems,readSingleton } from "@directus/sdk";
 | 
			
		||||
import directus from '../../lib/directus';
 | 
			
		||||
import { readItems, readSingleton } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
const global = await directus.request(readSingleton("global"));
 | 
			
		||||
const global = await directus.request(readSingleton('global'));
 | 
			
		||||
 | 
			
		||||
const posts = await directus.request(
 | 
			
		||||
  readItems("posts", {
 | 
			
		||||
  readItems('posts', {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
    sort: ["-published_date"],
 | 
			
		||||
    sort: ['-published_date'],
 | 
			
		||||
  })
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
@@ -18,42 +18,67 @@ const recentPosts = posts
 | 
			
		||||
  .sort((a, b) => b.published_date.getTime() - a.published_date.getTime())
 | 
			
		||||
  .slice(0, 3);
 | 
			
		||||
 | 
			
		||||
const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5);
 | 
			
		||||
 | 
			
		||||
const allTags = [...new Set(posts.flatMap((post) => post.tags || []))].slice(0, 5);
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<Layout title=`Home | ${global.name}`>
 | 
			
		||||
  <!-- Hero Section with improved mobile responsiveness -->
 | 
			
		||||
  <section class="py-10 sm:py-16 md:py-20 px-4 sm:px-6 theme-transition-all">
 | 
			
		||||
    <div class="max-w-2xl mx-auto relative">
 | 
			
		||||
  <section class="theme-transition-all px-4 py-10 sm:px-6 sm:py-16 md:py-20">
 | 
			
		||||
    <div class="relative mx-auto max-w-2xl">
 | 
			
		||||
      <!-- Adjusted blob positions and sizes for better mobile appearance -->
 | 
			
		||||
      <div class="absolute -top-10 sm:-top-20 -left-10 sm:-left-20 w-40 sm:w-64 h-40 sm:h-64 bg-zinc-100 dark:bg-zinc-800/50 rounded-full blur-3xl opacity-50 animate-blob theme-transition-bg"></div>
 | 
			
		||||
      <div class="absolute -bottom-10 sm:-bottom-20 -right-10 sm:-right-20 w-40 sm:w-64 h-40 sm:h-64 bg-zinc-200 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob animation-delay-2000 theme-transition-bg"></div>
 | 
			
		||||
      
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob theme-transition-bg absolute -left-10 -top-10 h-40 w-40 rounded-full bg-zinc-100 opacity-50 blur-3xl dark:bg-zinc-800/50 sm:-left-20 sm:-top-20 sm:h-64 sm:w-64"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 theme-transition-bg absolute -bottom-10 -right-10 h-40 w-40 rounded-full bg-zinc-200 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:-bottom-20 sm:-right-20 sm:h-64 sm:w-64"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="relative text-center sm:text-left">
 | 
			
		||||
        <h1 class="text-3xl sm:text-4xl md:text-5xl lg:text-6xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 theme-transition-color hero-text">
 | 
			
		||||
        <h1
 | 
			
		||||
          class="theme-transition-color hero-text text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:text-4xl md:text-5xl lg:text-6xl"
 | 
			
		||||
        >
 | 
			
		||||
          <span class="block">Writing on technology,</span>
 | 
			
		||||
          <span class="block mt-1">development, and</span>
 | 
			
		||||
          <span class="block mt-1 relative">
 | 
			
		||||
          <span class="mt-1 block">development, and</span>
 | 
			
		||||
          <span class="relative mt-1 block">
 | 
			
		||||
            <span class="relative inline-block">
 | 
			
		||||
              selfhosting.
 | 
			
		||||
              <span class="absolute -bottom-1 left-0 w-full h-1 bg-zinc-800 dark:bg-zinc-200 transform origin-left theme-transition-bg"></span>
 | 
			
		||||
              <span
 | 
			
		||||
                class="theme-transition-bg absolute -bottom-1 left-0 h-1 w-full origin-left transform bg-zinc-800 dark:bg-zinc-200"
 | 
			
		||||
              ></span>
 | 
			
		||||
            </span>
 | 
			
		||||
          </span>
 | 
			
		||||
        </h1>
 | 
			
		||||
        <p class="mt-4 sm:mt-6 md:mt-8 text-base sm:text-lg text-zinc-600 dark:text-zinc-400 leading-relaxed theme-transition-color max-w-lg mx-auto sm:mx-0">
 | 
			
		||||
        <p
 | 
			
		||||
          class="theme-transition-color mx-auto mt-4 max-w-lg text-base leading-relaxed text-zinc-600 dark:text-zinc-400 sm:mx-0 sm:mt-6 sm:text-lg md:mt-8"
 | 
			
		||||
        >
 | 
			
		||||
          {global.about}
 | 
			
		||||
        </p>
 | 
			
		||||
        <div class="mt-6 sm:mt-8 md:mt-10 flex flex-wrap gap-3 sm:gap-4 md:gap-6 justify-center sm:justify-start">
 | 
			
		||||
          <a 
 | 
			
		||||
            href="/about" 
 | 
			
		||||
            class="group relative inline-flex items-center gap-2 text-sm font-medium text-zinc-900 dark:text-zinc-100 hover:text-zinc-700 dark:hover:text-zinc-300 transition-all duration-300 theme-transition-color min-h-[44px]"
 | 
			
		||||
        <div
 | 
			
		||||
          class="mt-6 flex flex-wrap justify-center gap-3 sm:mt-8 sm:justify-start sm:gap-4 md:mt-10 md:gap-6"
 | 
			
		||||
        >
 | 
			
		||||
          <a
 | 
			
		||||
            href="/about"
 | 
			
		||||
            class="theme-transition-color group relative inline-flex min-h-[44px] items-center gap-2 text-sm font-medium text-zinc-900 transition-all duration-300 hover:text-zinc-700 dark:text-zinc-100 dark:hover:text-zinc-300"
 | 
			
		||||
          >
 | 
			
		||||
            <span>More about me</span>
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3" />
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="h-4 w-4 transition-transform duration-300 group-hover:translate-x-1"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3"></path>
 | 
			
		||||
            </svg>
 | 
			
		||||
            <span class="absolute -bottom-1 left-0 w-0 h-0.5 bg-zinc-800 dark:bg-zinc-200 transition-all duration-300 group-hover:w-full theme-transition-bg"></span>
 | 
			
		||||
            <span
 | 
			
		||||
              class="theme-transition-bg absolute -bottom-1 left-0 h-0.5 w-0 bg-zinc-800 transition-all duration-300 group-hover:w-full dark:bg-zinc-200"
 | 
			
		||||
            ></span>
 | 
			
		||||
          </a>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
@@ -61,139 +86,194 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
  </section>
 | 
			
		||||
 | 
			
		||||
  <!-- Featured Post Section - Improved for mobile -->
 | 
			
		||||
  <section class="py-10 sm:py-12 md:py-16 px-4 sm:px-6 border-t border-zinc-100 dark:border-zinc-800 theme-transition-all">
 | 
			
		||||
    <div class="max-w-3xl mx-auto">
 | 
			
		||||
      <div class="flex flex-col sm:flex-row sm:items-center justify-between gap-4 mb-6 sm:mb-8 md:mb-12">
 | 
			
		||||
        <h2 class="text-xl sm:text-2xl md:text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 theme-transition-color text-center sm:text-left">Recent Posts</h2>
 | 
			
		||||
        <a 
 | 
			
		||||
          href="/blog" 
 | 
			
		||||
          class="group relative text-sm font-medium text-zinc-900 dark:text-zinc-100 hover:text-zinc-700 dark:hover:text-zinc-300 self-center sm:self-auto theme-transition-color min-h-[44px] flex items-center justify-center"
 | 
			
		||||
  <section
 | 
			
		||||
    class="theme-transition-all border-t border-zinc-100 px-4 py-10 dark:border-zinc-800 sm:px-6 sm:py-12 md:py-16"
 | 
			
		||||
  >
 | 
			
		||||
    <div class="mx-auto max-w-3xl">
 | 
			
		||||
      <div
 | 
			
		||||
        class="mb-6 flex flex-col justify-between gap-4 sm:mb-8 sm:flex-row sm:items-center md:mb-12"
 | 
			
		||||
      >
 | 
			
		||||
        <h2
 | 
			
		||||
          class="theme-transition-color text-center text-xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:text-left sm:text-2xl md:text-3xl"
 | 
			
		||||
        >
 | 
			
		||||
          Recent Posts
 | 
			
		||||
        </h2>
 | 
			
		||||
        <a
 | 
			
		||||
          href="/blog"
 | 
			
		||||
          class="theme-transition-color group relative flex min-h-[44px] items-center justify-center self-center text-sm font-medium text-zinc-900 hover:text-zinc-700 dark:text-zinc-100 dark:hover:text-zinc-300 sm:self-auto"
 | 
			
		||||
        >
 | 
			
		||||
          <span class="flex items-center gap-1">
 | 
			
		||||
            View all posts
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3" />
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="h-4 w-4 transition-transform duration-300 group-hover:translate-x-1"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3"></path>
 | 
			
		||||
            </svg>
 | 
			
		||||
          </span>
 | 
			
		||||
          <span class="absolute -bottom-1 left-0 w-0 h-0.5 bg-zinc-800 dark:bg-zinc-200 transition-all duration-300 group-hover:w-full theme-transition-bg"></span>
 | 
			
		||||
          <span
 | 
			
		||||
            class="theme-transition-bg absolute -bottom-1 left-0 h-0.5 w-0 bg-zinc-800 transition-all duration-300 group-hover:w-full dark:bg-zinc-200"
 | 
			
		||||
          ></span>
 | 
			
		||||
        </a>
 | 
			
		||||
      </div>
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      <!-- Improved grid for better mobile layout -->
 | 
			
		||||
      <div class="grid gap-6 sm:gap-8 md:gap-12 grid-cols-1 sm:grid-cols-2 lg:grid-cols-3">
 | 
			
		||||
        {recentPosts.map((post, index) => (
 | 
			
		||||
          <article class="group relative flex flex-col items-start hover-3d theme-transition-element max-w-sm mx-auto sm:mx-0 w-full">
 | 
			
		||||
            <div class="absolute -inset-x-4 -inset-y-6 z-0 scale-95 bg-zinc-50 dark:bg-zinc-800/50 opacity-0 transition group-hover:scale-100 group-hover:opacity-100 sm:-inset-x-6 sm:rounded-2xl theme-transition-bg"></div>
 | 
			
		||||
            
 | 
			
		||||
            {post.image && (
 | 
			
		||||
              <div class="relative z-10 w-full aspect-video mb-4 overflow-hidden rounded-lg">
 | 
			
		||||
                <img 
 | 
			
		||||
                  src={`${process.env.DIRECTUS_URL ?? "https://directus.alexlebens.dev"}/assets/${post.image}`}
 | 
			
		||||
                  alt={post.title}
 | 
			
		||||
                  class="w-full h-full object-cover transition-transform duration-500 group-hover:scale-105"
 | 
			
		||||
                  loading={index === 0 ? "eager" : "lazy"}
 | 
			
		||||
                  width="400"
 | 
			
		||||
                  height="225"
 | 
			
		||||
                />
 | 
			
		||||
      <div class="grid grid-cols-1 gap-6 sm:grid-cols-2 sm:gap-8 md:gap-12 lg:grid-cols-3">
 | 
			
		||||
        {
 | 
			
		||||
          recentPosts.map((post, index) => (
 | 
			
		||||
            <article class="hover-3d theme-transition-element group relative mx-auto flex w-full max-w-sm flex-col items-start sm:mx-0">
 | 
			
		||||
              <div class="theme-transition-bg absolute -inset-x-4 -inset-y-6 z-0 scale-95 bg-zinc-50 opacity-0 transition group-hover:scale-100 group-hover:opacity-100 dark:bg-zinc-800/50 sm:-inset-x-6 sm:rounded-2xl" />
 | 
			
		||||
 | 
			
		||||
              {post.image && (
 | 
			
		||||
                <div class="relative z-10 mb-4 aspect-video w-full overflow-hidden rounded-lg">
 | 
			
		||||
                  <img
 | 
			
		||||
                    src={`${process.env.DIRECTUS_URL ?? 'https://directus.alexlebens.dev'}/assets/${post.image}`}
 | 
			
		||||
                    alt={post.title}
 | 
			
		||||
                    class="h-full w-full object-cover transition-transform duration-500 group-hover:scale-105"
 | 
			
		||||
                    loading={index === 0 ? 'eager' : 'lazy'}
 | 
			
		||||
                    width="400"
 | 
			
		||||
                    height="225"
 | 
			
		||||
                  />
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              <div class="theme-transition-color relative z-10 flex w-full flex-wrap items-center justify-center gap-x-3 gap-y-2 text-xs text-zinc-500 dark:text-zinc-400 sm:justify-start sm:gap-x-4">
 | 
			
		||||
                <time datetime={post.published_date.toLocaleString()} class="font-medium">
 | 
			
		||||
                  <FormattedDate date={post.published_date} />
 | 
			
		||||
                </time>
 | 
			
		||||
              </div>
 | 
			
		||||
            )}
 | 
			
		||||
            
 | 
			
		||||
            <div class="relative z-10 flex items-center flex-wrap gap-x-3 sm:gap-x-4 gap-y-2 text-xs text-zinc-500 dark:text-zinc-400 theme-transition-color justify-center sm:justify-start w-full">
 | 
			
		||||
              <time datetime={post.published_date.toLocaleString()} class="font-medium">
 | 
			
		||||
                <FormattedDate date={post.published_date} />
 | 
			
		||||
              </time>
 | 
			
		||||
            </div>
 | 
			
		||||
            
 | 
			
		||||
            <h3 class="relative z-10 mt-3 text-lg sm:text-xl font-semibold tracking-tight text-zinc-900 dark:text-zinc-100 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors theme-transition-color text-center sm:text-left w-full">
 | 
			
		||||
              <a href={`/blog/${post.slug}`} class="min-h-[44px] flex items-center justify-center sm:justify-start">
 | 
			
		||||
                <span class="absolute -inset-x-4 -inset-y-2.5 sm:-inset-x-6 sm:-inset-y-4"></span>
 | 
			
		||||
                {post.title}
 | 
			
		||||
              </a>
 | 
			
		||||
            </h3>
 | 
			
		||||
            
 | 
			
		||||
            <p class="relative z-10 mt-2 sm:mt-3 text-sm text-zinc-600 dark:text-zinc-400 line-clamp-3 theme-transition-color text-center sm:text-left w-full">
 | 
			
		||||
              {post.description}
 | 
			
		||||
            </p>
 | 
			
		||||
            
 | 
			
		||||
            {post.tags && post.tags.length > 0 && (
 | 
			
		||||
              <div class="relative z-10 mt-3 sm:mt-4 flex flex-wrap gap-2 justify-center sm:justify-start w-full">
 | 
			
		||||
                {post.tags.slice(0, 3).map(tag => (
 | 
			
		||||
                  <a 
 | 
			
		||||
                    href={`/topics/${tag}`}
 | 
			
		||||
                    class="inline-flex items-center rounded-full bg-zinc-100 px-2 sm:px-3 py-1 text-xs font-medium text-zinc-800 hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-100 dark:hover:bg-zinc-700 transition-colors theme-transition-all min-h-[28px]"
 | 
			
		||||
                  >
 | 
			
		||||
                    #{tag}
 | 
			
		||||
                  </a>
 | 
			
		||||
                ))}
 | 
			
		||||
                {post.tags.length > 3 && (
 | 
			
		||||
                  <span class="inline-flex items-center rounded-full bg-zinc-50 px-2 py-1 text-xs font-medium text-zinc-500 dark:bg-zinc-800/50 dark:text-zinc-400 theme-transition-all min-h-[28px]">
 | 
			
		||||
                    +{post.tags.length - 3} more
 | 
			
		||||
 | 
			
		||||
              <h3 class="theme-transition-color relative z-10 mt-3 w-full text-center text-lg font-semibold tracking-tight text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-zinc-100 dark:group-hover:text-zinc-300 sm:text-left sm:text-xl">
 | 
			
		||||
                <a
 | 
			
		||||
                  href={`/blog/${post.slug}`}
 | 
			
		||||
                  class="flex min-h-[44px] items-center justify-center sm:justify-start"
 | 
			
		||||
                >
 | 
			
		||||
                  <span class="absolute -inset-x-4 -inset-y-2.5 sm:-inset-x-6 sm:-inset-y-4" />
 | 
			
		||||
                  {post.title}
 | 
			
		||||
                </a>
 | 
			
		||||
              </h3>
 | 
			
		||||
 | 
			
		||||
              <p class="theme-transition-color relative z-10 mt-2 line-clamp-3 w-full text-center text-sm text-zinc-600 dark:text-zinc-400 sm:mt-3 sm:text-left">
 | 
			
		||||
                {post.description}
 | 
			
		||||
              </p>
 | 
			
		||||
 | 
			
		||||
              {post.tags && post.tags.length > 0 && (
 | 
			
		||||
                <div class="relative z-10 mt-3 flex w-full flex-wrap justify-center gap-2 sm:mt-4 sm:justify-start">
 | 
			
		||||
                  {post.tags.slice(0, 3).map((tag) => (
 | 
			
		||||
                    <a
 | 
			
		||||
                      href={`/topics/${tag}`}
 | 
			
		||||
                      class="theme-transition-all inline-flex min-h-[28px] items-center rounded-full bg-zinc-100 px-2 py-1 text-xs font-medium text-zinc-800 transition-colors hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-100 dark:hover:bg-zinc-700 sm:px-3"
 | 
			
		||||
                    >
 | 
			
		||||
                      #{tag}
 | 
			
		||||
                    </a>
 | 
			
		||||
                  ))}
 | 
			
		||||
                  {post.tags.length > 3 && (
 | 
			
		||||
                    <span class="theme-transition-all inline-flex min-h-[28px] items-center rounded-full bg-zinc-50 px-2 py-1 text-xs font-medium text-zinc-500 dark:bg-zinc-800/50 dark:text-zinc-400">
 | 
			
		||||
                      +{post.tags.length - 3} more
 | 
			
		||||
                    </span>
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              <a
 | 
			
		||||
                href={`/blog/${post.slug}`}
 | 
			
		||||
                class="theme-transition-color relative z-10 mx-auto mt-3 flex min-h-[44px] items-center text-sm font-medium text-zinc-700 transition-colors group-hover:text-zinc-900 dark:text-zinc-300 dark:group-hover:text-zinc-100 sm:mx-0 sm:mt-4"
 | 
			
		||||
              >
 | 
			
		||||
                <span class="relative inline-block overflow-hidden">
 | 
			
		||||
                  <span class="block transition-transform duration-300 group-hover:-translate-y-full">
 | 
			
		||||
                    Read article
 | 
			
		||||
                  </span>
 | 
			
		||||
                )}
 | 
			
		||||
              </div>
 | 
			
		||||
            )}
 | 
			
		||||
            
 | 
			
		||||
            <a 
 | 
			
		||||
              href={`/blog/${post.slug}`}
 | 
			
		||||
              class="relative z-10 mt-3 sm:mt-4 flex items-center text-sm font-medium text-zinc-700 dark:text-zinc-300 group-hover:text-zinc-900 dark:group-hover:text-zinc-100 transition-colors theme-transition-color mx-auto sm:mx-0 min-h-[44px]"
 | 
			
		||||
            >
 | 
			
		||||
              <span class="relative overflow-hidden inline-block">
 | 
			
		||||
                <span class="group-hover:-translate-y-full block transition-transform duration-300">Read article</span>
 | 
			
		||||
                <span class="absolute top-0 left-0 translate-y-full group-hover:translate-y-0 transition-transform duration-300 whitespace-nowrap">Explore now</span>
 | 
			
		||||
              </span>
 | 
			
		||||
              <svg viewBox="0 0 16 16" fill="none" aria-hidden="true" class="ml-1 h-4 w-4 stroke-current transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
                <path d="M6.75 5.75 9.25 8l-2.5 2.25" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"></path>
 | 
			
		||||
              </svg>
 | 
			
		||||
            </a>
 | 
			
		||||
          </article>
 | 
			
		||||
        ))}
 | 
			
		||||
                  <span class="absolute left-0 top-0 translate-y-full whitespace-nowrap transition-transform duration-300 group-hover:translate-y-0">
 | 
			
		||||
                    Explore now
 | 
			
		||||
                  </span>
 | 
			
		||||
                </span>
 | 
			
		||||
                <svg
 | 
			
		||||
                  viewBox="0 0 16 16"
 | 
			
		||||
                  fill="none"
 | 
			
		||||
                  aria-hidden="true"
 | 
			
		||||
                  class="ml-1 h-4 w-4 stroke-current transition-transform duration-300 group-hover:translate-x-1"
 | 
			
		||||
                >
 | 
			
		||||
                  <path
 | 
			
		||||
                    d="M6.75 5.75 9.25 8l-2.5 2.25"
 | 
			
		||||
                    stroke-width="1.5"
 | 
			
		||||
                    stroke-linecap="round"
 | 
			
		||||
                    stroke-linejoin="round"
 | 
			
		||||
                  />
 | 
			
		||||
                </svg>
 | 
			
		||||
              </a>
 | 
			
		||||
            </article>
 | 
			
		||||
          ))
 | 
			
		||||
        }
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  </section>
 | 
			
		||||
 | 
			
		||||
  <!-- Topics/Tags Section - Improved for mobile -->
 | 
			
		||||
  {allTags.length > 0 && (
 | 
			
		||||
    <section class="py-10 sm:py-12 md:py-16 px-4 sm:px-6 border-t border-zinc-100 dark:border-zinc-800 theme-transition-all">
 | 
			
		||||
      <div class="max-w-3xl mx-auto">
 | 
			
		||||
        <h2 class="text-xl sm:text-2xl md:text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 mb-6 sm:mb-8 theme-transition-color text-center sm:text-left">Explore Topics</h2>
 | 
			
		||||
        
 | 
			
		||||
        <!-- Improved grid layout for mobile -->
 | 
			
		||||
        <div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-3 sm:gap-4 max-w-xs sm:max-w-none mx-auto">
 | 
			
		||||
          {allTags.map(tag => {
 | 
			
		||||
            const tagCount = posts.filter(post => post.tags && post.tags.includes(tag)).length;
 | 
			
		||||
            return (
 | 
			
		||||
              <a 
 | 
			
		||||
                href={`/topics/${tag}`}
 | 
			
		||||
                class="group flex flex-col p-3 sm:p-4 md:p-6 rounded-xl border border-zinc-200 dark:border-zinc-800 hover:bg-zinc-50 dark:hover:bg-zinc-800/70 transition-all duration-300 theme-transition-all min-h-[80px] sm:min-h-[90px]"
 | 
			
		||||
  {
 | 
			
		||||
    allTags.length > 0 && (
 | 
			
		||||
      <section class="theme-transition-all border-t border-zinc-100 px-4 py-10 dark:border-zinc-800 sm:px-6 sm:py-12 md:py-16">
 | 
			
		||||
        <div class="mx-auto max-w-3xl">
 | 
			
		||||
          <h2 class="theme-transition-color mb-6 text-center text-xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:mb-8 sm:text-left sm:text-2xl md:text-3xl">
 | 
			
		||||
            Explore Topics
 | 
			
		||||
          </h2>
 | 
			
		||||
 | 
			
		||||
          <div class="mx-auto grid max-w-xs grid-cols-1 gap-3 sm:max-w-none sm:grid-cols-2 sm:gap-4 md:grid-cols-3">
 | 
			
		||||
            {allTags.map((tag) => {
 | 
			
		||||
              const tagCount = posts.filter((post) => post.tags && post.tags.includes(tag)).length;
 | 
			
		||||
              return (
 | 
			
		||||
                <a
 | 
			
		||||
                  href={`/topics/${tag}`}
 | 
			
		||||
                  class="theme-transition-all group flex min-h-[80px] flex-col rounded-xl border border-zinc-200 p-3 transition-all duration-300 hover:bg-zinc-50 dark:border-zinc-800 dark:hover:bg-zinc-800/70 sm:min-h-[90px] sm:p-4 md:p-6"
 | 
			
		||||
                >
 | 
			
		||||
                  <div class="mb-2 flex items-start justify-between">
 | 
			
		||||
                    <span class="theme-transition-color mr-2 text-sm font-medium text-zinc-900 dark:text-zinc-100">
 | 
			
		||||
                      #{tag}
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <span class="theme-transition-all flex-shrink-0 rounded-full bg-zinc-100 px-2 py-0.5 text-xs text-zinc-500 dark:bg-zinc-800 dark:text-zinc-400">
 | 
			
		||||
                      {tagCount} {tagCount === 1 ? 'post' : 'posts'}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <p class="theme-transition-color mt-1 text-xs text-zinc-600 dark:text-zinc-400">
 | 
			
		||||
                    Explore articles about {tag}
 | 
			
		||||
                  </p>
 | 
			
		||||
                </a>
 | 
			
		||||
              );
 | 
			
		||||
            })}
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <div class="mt-6 text-center sm:mt-8">
 | 
			
		||||
            <a
 | 
			
		||||
              href="/tags"
 | 
			
		||||
              class="theme-transition-color inline-flex min-h-[44px] items-center text-sm font-medium text-zinc-900 hover:text-zinc-700 dark:text-zinc-100 dark:hover:text-zinc-300"
 | 
			
		||||
            >
 | 
			
		||||
              <span>View all topics</span>
 | 
			
		||||
              <svg
 | 
			
		||||
                xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                fill="none"
 | 
			
		||||
                viewBox="0 0 24 24"
 | 
			
		||||
                stroke-width="1.5"
 | 
			
		||||
                stroke="currentColor"
 | 
			
		||||
                class="ml-1 h-4 w-4 transition-transform duration-300 group-hover:translate-x-1"
 | 
			
		||||
              >
 | 
			
		||||
                <div class="flex items-start justify-between mb-2">
 | 
			
		||||
                  <span class="text-sm font-medium text-zinc-900 dark:text-zinc-100 theme-transition-color mr-2">#{tag}</span>
 | 
			
		||||
                  <span class="text-xs bg-zinc-100 dark:bg-zinc-800 text-zinc-500 dark:text-zinc-400 px-2 py-0.5 rounded-full flex-shrink-0 theme-transition-all">
 | 
			
		||||
                    {tagCount} {tagCount === 1 ? 'post' : 'posts'}
 | 
			
		||||
                  </span>
 | 
			
		||||
                </div>
 | 
			
		||||
                <p class="text-xs text-zinc-600 dark:text-zinc-400 mt-1 theme-transition-color">
 | 
			
		||||
                  Explore articles about {tag}
 | 
			
		||||
                </p>
 | 
			
		||||
              </a>
 | 
			
		||||
            )
 | 
			
		||||
          })}
 | 
			
		||||
                <path
 | 
			
		||||
                  stroke-linecap="round"
 | 
			
		||||
                  stroke-linejoin="round"
 | 
			
		||||
                  d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3"
 | 
			
		||||
                />
 | 
			
		||||
              </svg>
 | 
			
		||||
            </a>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        
 | 
			
		||||
        <div class="mt-6 sm:mt-8 text-center">
 | 
			
		||||
          <a 
 | 
			
		||||
            href="/tags" 
 | 
			
		||||
            class="inline-flex items-center text-sm font-medium text-zinc-900 dark:text-zinc-100 hover:text-zinc-700 dark:hover:text-zinc-300 theme-transition-color min-h-[44px]"
 | 
			
		||||
          >
 | 
			
		||||
            <span>View all topics</span>
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 ml-1 transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </a>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </section>
 | 
			
		||||
  )}
 | 
			
		||||
      </section>
 | 
			
		||||
    )
 | 
			
		||||
  }
 | 
			
		||||
</Layout>
 | 
			
		||||
 | 
			
		||||
<script>
 | 
			
		||||
@@ -201,69 +281,69 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', () => {
 | 
			
		||||
    // Check if it's a touch device
 | 
			
		||||
    const isTouchDevice = 'ontouchstart' in window || navigator.maxTouchPoints > 0;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    if (isTouchDevice) {
 | 
			
		||||
      const cards = document.querySelectorAll('.hover-3d');
 | 
			
		||||
      
 | 
			
		||||
      cards.forEach(card => {
 | 
			
		||||
 | 
			
		||||
      cards.forEach((card) => {
 | 
			
		||||
        card.addEventListener('touchstart', () => {
 | 
			
		||||
          card.classList.add('is-touched');
 | 
			
		||||
        });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        card.addEventListener('touchend', () => {
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            card.classList.remove('is-touched');
 | 
			
		||||
          }, 300);
 | 
			
		||||
        });
 | 
			
		||||
      });
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Disable hover animations on touch devices for better performance
 | 
			
		||||
      document.documentElement.classList.add('touch-device');
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Improved viewport height fix for mobile browsers
 | 
			
		||||
    const setVh = () => {
 | 
			
		||||
      const vh = window.innerHeight * 0.01;
 | 
			
		||||
      document.documentElement.style.setProperty('--vh', `${vh}px`);
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Set initial value
 | 
			
		||||
    setVh();
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Update on resize and scroll to prevent content shifting
 | 
			
		||||
    window.addEventListener('resize', setVh);
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Use a debounced scroll handler to prevent performance issues
 | 
			
		||||
    let scrollTimeout;
 | 
			
		||||
    window.addEventListener('scroll', () => {
 | 
			
		||||
      if (scrollTimeout) {
 | 
			
		||||
        window.cancelAnimationFrame(scrollTimeout);
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      scrollTimeout = window.requestAnimationFrame(() => {
 | 
			
		||||
        // Lock width during scroll
 | 
			
		||||
        document.body.style.width = '100%';
 | 
			
		||||
        document.body.style.overflowX = 'hidden';
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Fix for iOS Safari address bar height changes
 | 
			
		||||
    if (/iPhone|iPad|iPod/.test(navigator.userAgent)) {
 | 
			
		||||
      // Force the layout to use the initial viewport size
 | 
			
		||||
      document.documentElement.style.setProperty('--initial-vh', `${window.innerHeight * 0.01}px`);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Apply fixed height to sections to prevent resizing
 | 
			
		||||
      const sections = document.querySelectorAll('section');
 | 
			
		||||
      sections.forEach(section => {
 | 
			
		||||
      sections.forEach((section) => {
 | 
			
		||||
        section.style.width = '100%';
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Improved theme change handler that preserves scroll position and provides smoother transitions
 | 
			
		||||
    document.addEventListener('themeChanged', () => {
 | 
			
		||||
      // Store current scroll position
 | 
			
		||||
      const scrollPosition = window.scrollY;
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Create a temporary overlay for smoother transition
 | 
			
		||||
      const overlay = document.createElement('div');
 | 
			
		||||
      overlay.style.cssText = `
 | 
			
		||||
@@ -276,19 +356,22 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
        transition: opacity 0.3s ease;
 | 
			
		||||
      `;
 | 
			
		||||
      document.body.appendChild(overlay);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Fade in overlay
 | 
			
		||||
      requestAnimationFrame(() => {
 | 
			
		||||
        overlay.style.opacity = '0.5';
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Update theme-transition elements without forcing reflow of entire page
 | 
			
		||||
        requestAnimationFrame(() => {
 | 
			
		||||
          document.querySelectorAll('.theme-transition-all, .theme-transition-bg, .theme-transition-color')
 | 
			
		||||
            .forEach(el => {
 | 
			
		||||
          document
 | 
			
		||||
            .querySelectorAll(
 | 
			
		||||
              '.theme-transition-all, .theme-transition-bg, .theme-transition-color'
 | 
			
		||||
            )
 | 
			
		||||
            .forEach((el) => {
 | 
			
		||||
              // Apply a subtle animation instead of a hard reset
 | 
			
		||||
              el.style.transition = 'all 0.5s ease';
 | 
			
		||||
            });
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Fade out overlay after transition completes
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            overlay.style.opacity = '0';
 | 
			
		||||
@@ -298,24 +381,24 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
          }, 300);
 | 
			
		||||
        });
 | 
			
		||||
      });
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Restore scroll position (prevents jumping to top)
 | 
			
		||||
      if (scrollPosition > 0) {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          window.scrollTo({
 | 
			
		||||
            top: scrollPosition,
 | 
			
		||||
            behavior: 'auto' // Use 'auto' to prevent animation
 | 
			
		||||
            behavior: 'auto', // Use 'auto' to prevent animation
 | 
			
		||||
          });
 | 
			
		||||
        }, 10);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Fix theme inconsistency issues by checking theme on visibility change
 | 
			
		||||
    document.addEventListener('visibilitychange', () => {
 | 
			
		||||
      if (document.visibilityState === 'visible') {
 | 
			
		||||
        const storedTheme = localStorage.getItem('theme');
 | 
			
		||||
        const currentThemeIsDark = document.documentElement.classList.contains('dark');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        if (storedTheme === 'dark' && !currentThemeIsDark) {
 | 
			
		||||
          document.documentElement.classList.add('dark');
 | 
			
		||||
        } else if (storedTheme === 'light' && currentThemeIsDark) {
 | 
			
		||||
@@ -323,34 +406,45 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Add smooth reveal animations for content after loading
 | 
			
		||||
    const animateContent = () => {
 | 
			
		||||
      // Animate hero section
 | 
			
		||||
      const heroElements = document.querySelectorAll('.hero-text span, .hero-text + p, .hero-text ~ div');
 | 
			
		||||
      const heroElements = document.querySelectorAll(
 | 
			
		||||
        '.hero-text span, .hero-text + p, .hero-text ~ div'
 | 
			
		||||
      );
 | 
			
		||||
      heroElements.forEach((el, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          el.classList.add('animate-reveal');
 | 
			
		||||
        }, 100 + (index * 150));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            el.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          100 + index * 150
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Animate posts with staggered delay
 | 
			
		||||
      const articles = document.querySelectorAll('article.group');
 | 
			
		||||
      articles.forEach((article, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          article.classList.add('animate-reveal');
 | 
			
		||||
        }, 500 + (index * 150));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            article.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          500 + index * 150
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Animate topic cards with staggered delay
 | 
			
		||||
      const topicCards = document.querySelectorAll('a.group.flex.flex-col');
 | 
			
		||||
      topicCards.forEach((card, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          card.classList.add('animate-reveal');
 | 
			
		||||
        }, 800 + (index * 100));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            card.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          800 + index * 100
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Run animations after the loading screen is hidden
 | 
			
		||||
    const loadingScreen = document.getElementById('loading-screen');
 | 
			
		||||
    if (loadingScreen) {
 | 
			
		||||
@@ -361,18 +455,20 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
        // Wait for loading screen to hide
 | 
			
		||||
        const observer = new MutationObserver((mutations) => {
 | 
			
		||||
          mutations.forEach((mutation) => {
 | 
			
		||||
            if (mutation.target === loadingScreen && 
 | 
			
		||||
                mutation.type === 'attributes' && 
 | 
			
		||||
                mutation.attributeName === 'style' &&
 | 
			
		||||
                loadingScreen.style.display === 'none') {
 | 
			
		||||
            if (
 | 
			
		||||
              mutation.target === loadingScreen &&
 | 
			
		||||
              mutation.type === 'attributes' &&
 | 
			
		||||
              mutation.attributeName === 'style' &&
 | 
			
		||||
              loadingScreen.style.display === 'none'
 | 
			
		||||
            ) {
 | 
			
		||||
              animateContent();
 | 
			
		||||
              observer.disconnect();
 | 
			
		||||
            }
 | 
			
		||||
          });
 | 
			
		||||
        });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        observer.observe(loadingScreen, { attributes: true });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Fallback
 | 
			
		||||
        setTimeout(animateContent, 3500);
 | 
			
		||||
      }
 | 
			
		||||
@@ -381,36 +477,38 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
      animateContent();
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // SPA transition handling for homepage
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all internal links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links, external links, or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') || 
 | 
			
		||||
          link.getAttribute('target') === '_blank' || 
 | 
			
		||||
          link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
      if (
 | 
			
		||||
        link.getAttribute('href').includes('#') ||
 | 
			
		||||
        link.getAttribute('target') === '_blank' ||
 | 
			
		||||
        link.hasAttribute('data-spa-handled')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Mark as handled to avoid duplicate listeners
 | 
			
		||||
      link.setAttribute('data-spa-handled', 'true');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      link.addEventListener('click', (e) => {
 | 
			
		||||
        // Don't handle if modifier keys are pressed (for opening in new tab, etc.)
 | 
			
		||||
        if (e.metaKey || e.ctrlKey || e.shiftKey || e.altKey) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetHref = link.getAttribute('href');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Trigger page transition animation
 | 
			
		||||
        const pageTransition = document.getElementById('page-transition');
 | 
			
		||||
        if (pageTransition) {
 | 
			
		||||
          pageTransition.classList.remove('opacity-0');
 | 
			
		||||
          pageTransition.classList.add('opacity-100');
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Navigate after transition effect
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            window.location.href = targetHref;
 | 
			
		||||
@@ -422,13 +520,13 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Initialize on first load
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Re-initialize when content changes via Astro's view transitions
 | 
			
		||||
  document.addEventListener('astro:page-load', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // For compatibility with custom transition system
 | 
			
		||||
  document.addEventListener('page-transition-complete', setupSPATransitions);
 | 
			
		||||
</script>
 | 
			
		||||
@@ -439,51 +537,55 @@ const allTags = [...new Set(posts.flatMap(post => post.tags || []))].slice(0, 5)
 | 
			
		||||
    --theme-transition-duration: 0.5s;
 | 
			
		||||
    --theme-transition-timing: ease;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
  :global(html), :global(body) {
 | 
			
		||||
 | 
			
		||||
  :global(html),
 | 
			
		||||
  :global(body) {
 | 
			
		||||
    transition: background-color var(--theme-transition-duration) var(--theme-transition-timing);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  :global(.theme-transition-all) {
 | 
			
		||||
    transition: all var(--theme-transition-duration) var(--theme-transition-timing);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  :global(.theme-transition-bg) {
 | 
			
		||||
    transition: background-color var(--theme-transition-duration) var(--theme-transition-timing);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  :global(.theme-transition-color) {
 | 
			
		||||
    transition: color var(--theme-transition-duration) var(--theme-transition-timing);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Ensure transitions apply to all theme-related properties */
 | 
			
		||||
  :global(*) {
 | 
			
		||||
    transition-property: background-color, border-color, color, fill, stroke, opacity;
 | 
			
		||||
    transition-duration: var(--theme-transition-duration);
 | 
			
		||||
    transition-timing-function: var(--theme-transition-timing);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Remove the forced transition disabling which causes flickering */
 | 
			
		||||
  :global(.theme-switching), :global(.theme-switching *) {
 | 
			
		||||
  :global(.theme-switching),
 | 
			
		||||
  :global(.theme-switching *) {
 | 
			
		||||
    /* Use a subtle transition instead of none */
 | 
			
		||||
    transition-duration: 0.3s !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Content reveal animations */
 | 
			
		||||
  .hero-text span, 
 | 
			
		||||
  .hero-text + p, 
 | 
			
		||||
  .hero-text span,
 | 
			
		||||
  .hero-text + p,
 | 
			
		||||
  .hero-text ~ div,
 | 
			
		||||
  article.group,
 | 
			
		||||
  a.group.flex.flex-col {
 | 
			
		||||
    opacity: 0;
 | 
			
		||||
    transform: translateY(20px);
 | 
			
		||||
    transition: opacity 0.8s ease, transform 0.8s ease;
 | 
			
		||||
    transition:
 | 
			
		||||
      opacity 0.8s ease,
 | 
			
		||||
      transform 0.8s ease;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animate-reveal {
 | 
			
		||||
    opacity: 1 !important;
 | 
			
		||||
    transform: translateY(0) !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Rest of your existing styles... */
 | 
			
		||||
</style>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +1,17 @@
 | 
			
		||||
import rss from '@astrojs/rss';
 | 
			
		||||
 | 
			
		||||
import directus from "../../lib/directus"
 | 
			
		||||
import { readItems,readSingleton } from "@directus/sdk";
 | 
			
		||||
import directus from '../../lib/directus';
 | 
			
		||||
import { readItems, readSingleton } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
export async function GET(context: any) {
 | 
			
		||||
  const global = await directus.request(readSingleton("global"));
 | 
			
		||||
  const global = await directus.request(readSingleton('global'));
 | 
			
		||||
  const posts = await directus.request(
 | 
			
		||||
    readItems("posts", {
 | 
			
		||||
    readItems('posts', {
 | 
			
		||||
      fields: ['*'],
 | 
			
		||||
      sort: ["-published_date"],
 | 
			
		||||
      sort: ['-published_date'],
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  return rss({
 | 
			
		||||
    title: `${global.name}`,
 | 
			
		||||
    description: `${global.description}`,
 | 
			
		||||
 
 | 
			
		||||
@@ -2,24 +2,26 @@
 | 
			
		||||
import BaseLayout from '../../layouts/BaseLayout.astro';
 | 
			
		||||
import FormattedDate from '../../components/FormattedDate.astro';
 | 
			
		||||
 | 
			
		||||
import directus from "../../../lib/directus"
 | 
			
		||||
import { readItems } from "@directus/sdk";
 | 
			
		||||
import directus from '../../../lib/directus';
 | 
			
		||||
import { readItems } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
export const prerender = true;
 | 
			
		||||
 | 
			
		||||
export async function getStaticPaths() {
 | 
			
		||||
  const posts = await directus.request(readItems("posts", {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
  }));
 | 
			
		||||
  const posts = await directus.request(
 | 
			
		||||
    readItems('posts', {
 | 
			
		||||
      fields: ['*'],
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  // Get all unique tags
 | 
			
		||||
  const uniqueTags = [...new Set(posts.flatMap(post => post.tags || []))];
 | 
			
		||||
  const uniqueTags = [...new Set(posts.flatMap((post) => post.tags || []))];
 | 
			
		||||
 | 
			
		||||
  // Create a path for each tag
 | 
			
		||||
  return uniqueTags.map(tag => {
 | 
			
		||||
  return uniqueTags.map((tag) => {
 | 
			
		||||
    // Make tag matching case-insensitive
 | 
			
		||||
    const filteredPosts = posts.filter(post =>
 | 
			
		||||
      post.tags?.some(t => t.toLowerCase() === (tag as string).toLowerCase()) // Explicitly cast tag to string
 | 
			
		||||
    const filteredPosts = posts.filter(
 | 
			
		||||
      (post) => post.tags?.some((t) => t.toLowerCase() === (tag as string).toLowerCase()) // Explicitly cast tag to string
 | 
			
		||||
    );
 | 
			
		||||
    return {
 | 
			
		||||
      params: { tag },
 | 
			
		||||
@@ -33,184 +35,295 @@ const { posts = [] } = Astro.props;
 | 
			
		||||
 | 
			
		||||
console.log(`Tag: ${tag}, Number of posts: ${posts.length}`);
 | 
			
		||||
 | 
			
		||||
const sortedPosts = posts && posts.length > 0
 | 
			
		||||
  ? [...posts].sort((a, b) => b.published_date.valueOf() - a.published_date.valueOf())
 | 
			
		||||
  : [];
 | 
			
		||||
const sortedPosts =
 | 
			
		||||
  posts && posts.length > 0
 | 
			
		||||
    ? [...posts].sort((a, b) => b.published_date.valueOf() - a.published_date.valueOf())
 | 
			
		||||
    : [];
 | 
			
		||||
console.log(`Sorted posts length: ${sortedPosts.length}`);
 | 
			
		||||
 | 
			
		||||
const tagHue = Math.abs(tag.split('').reduce((acc, char) => acc + char.charCodeAt(0), 0) % 360);
 | 
			
		||||
const relatedTags = [...new Set(
 | 
			
		||||
  sortedPosts.flatMap(post => post.tags || [])
 | 
			
		||||
    .filter(t => t !== tag)
 | 
			
		||||
)].slice(0, 5);
 | 
			
		||||
 | 
			
		||||
const relatedTags = [
 | 
			
		||||
  ...new Set(sortedPosts.flatMap((post) => post.tags || []).filter((t) => t !== tag)),
 | 
			
		||||
].slice(0, 5);
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<BaseLayout title={`Posts tagged with "${tag}"`}>
 | 
			
		||||
  <div class="max-w-5xl mx-auto px-4 py-10 sm:py-16">
 | 
			
		||||
  <div class="mx-auto max-w-5xl px-4 py-10 sm:py-16">
 | 
			
		||||
    <!-- Header section -->
 | 
			
		||||
    <div class="relative mb-10 sm:mb-16">
 | 
			
		||||
      <div class="absolute -top-20 -left-20 w-48 sm:w-64 h-48 sm:h-64 bg-zinc-100 dark:bg-zinc-900/30 rounded-full blur-3xl opacity-30 animate-blob"></div>
 | 
			
		||||
      <div class="absolute -bottom-10 -right-10 w-36 sm:w-48 h-36 sm:h-48 bg-zinc-200 dark:bg-zinc-900/20 rounded-full blur-2xl opacity-20 animate-blob animation-delay-2000"></div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob absolute -left-20 -top-20 h-48 w-48 rounded-full bg-zinc-100 opacity-30 blur-3xl dark:bg-zinc-900/30 sm:h-64 sm:w-64"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 absolute -bottom-10 -right-10 h-36 w-36 rounded-full bg-zinc-200 opacity-20 blur-2xl dark:bg-zinc-900/20 sm:h-48 sm:w-48"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="relative text-center sm:text-left">
 | 
			
		||||
        <a href="/tags" class="inline-flex items-center gap-2 text-sm font-medium text-zinc-600 dark:text-zinc-400 hover:text-zinc-900 dark:hover:text-zinc-100 transition-colors mb-4 group">
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 transition-transform duration-300 group-hover:-translate-x-1">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M10.5 19.5L3 12m0 0l7.5-7.5M3 12h18" />
 | 
			
		||||
        <a
 | 
			
		||||
          href="/tags"
 | 
			
		||||
          class="group mb-4 inline-flex items-center gap-2 text-sm font-medium text-zinc-600 transition-colors hover:text-zinc-900 dark:text-zinc-400 dark:hover:text-zinc-100"
 | 
			
		||||
        >
 | 
			
		||||
          <svg
 | 
			
		||||
            xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
            fill="none"
 | 
			
		||||
            viewBox="0 0 24 24"
 | 
			
		||||
            stroke-width="1.5"
 | 
			
		||||
            stroke="currentColor"
 | 
			
		||||
            class="h-4 w-4 transition-transform duration-300 group-hover:-translate-x-1"
 | 
			
		||||
          >
 | 
			
		||||
            <path
 | 
			
		||||
              stroke-linecap="round"
 | 
			
		||||
              stroke-linejoin="round"
 | 
			
		||||
              d="M10.5 19.5L3 12m0 0l7.5-7.5M3 12h18"></path>
 | 
			
		||||
          </svg>
 | 
			
		||||
          <span>Back to all topics</span>
 | 
			
		||||
          <span class="block max-w-0 group-hover:max-w-full transition-all duration-300 h-0.5 bg-zinc-300 dark:bg-zinc-700"></span>
 | 
			
		||||
          <span
 | 
			
		||||
            class="block h-0.5 max-w-0 bg-zinc-300 transition-all duration-300 group-hover:max-w-full dark:bg-zinc-700"
 | 
			
		||||
          ></span>
 | 
			
		||||
        </a>
 | 
			
		||||
 | 
			
		||||
        <div class="flex flex-col sm:flex-row sm:items-center gap-4 mb-2 justify-center sm:justify-start">
 | 
			
		||||
          <div class="tag-icon flex items-center justify-center w-12 h-12 rounded-xl bg-zinc-100 dark:bg-zinc-800 shadow-sm mx-auto sm:mx-0">
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6 text-zinc-700 dark:text-zinc-300">
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M9.568 3H5.25A2.25 2.25 0 003 5.25v4.318c0 .597.237 1.17.659 1.591l9.581 9.581c.699.699 1.78.872 2.607.33a18.095 18.095 0 005.223-5.223c.542-.827.369-1.908-.33-2.607L11.16 3.66A2.25 2.25 0 009.568 3z" />
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M6 6h.008v.008H6V6z" />
 | 
			
		||||
        <div
 | 
			
		||||
          class="mb-2 flex flex-col justify-center gap-4 sm:flex-row sm:items-center sm:justify-start"
 | 
			
		||||
        >
 | 
			
		||||
          <div
 | 
			
		||||
            class="tag-icon mx-auto flex h-12 w-12 items-center justify-center rounded-xl bg-zinc-100 shadow-sm dark:bg-zinc-800 sm:mx-0"
 | 
			
		||||
          >
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="h-6 w-6 text-zinc-700 dark:text-zinc-300"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M9.568 3H5.25A2.25 2.25 0 003 5.25v4.318c0 .597.237 1.17.659 1.591l9.581 9.581c.699.699 1.78.872 2.607.33a18.095 18.095 0 005.223-5.223c.542-.827.369-1.908-.33-2.607L11.16 3.66A2.25 2.25 0 009.568 3z"
 | 
			
		||||
              ></path>
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M6 6h.008v.008H6V6z"></path>
 | 
			
		||||
            </svg>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <h1 class="text-3xl sm:text-4xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100">
 | 
			
		||||
          <h1
 | 
			
		||||
            class="text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:text-4xl"
 | 
			
		||||
          >
 | 
			
		||||
            <span class="relative">
 | 
			
		||||
              #{tag}
 | 
			
		||||
              <span class="absolute -bottom-1 left-0 w-full h-1 bg-zinc-200 dark:bg-zinc-700"></span>
 | 
			
		||||
              <span class="absolute -bottom-1 left-0 w-1/2 h-1 bg-zinc-900 dark:bg-zinc-100 opacity-70 animate-expand"></span>
 | 
			
		||||
              <span class="absolute -bottom-1 left-0 h-1 w-full bg-zinc-200 dark:bg-zinc-700"
 | 
			
		||||
              ></span>
 | 
			
		||||
              <span
 | 
			
		||||
                class="animate-expand absolute -bottom-1 left-0 h-1 w-1/2 bg-zinc-900 opacity-70 dark:bg-zinc-100"
 | 
			
		||||
              ></span>
 | 
			
		||||
            </span>
 | 
			
		||||
          </h1>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <p class="text-base sm:text-lg text-zinc-600 dark:text-zinc-400 mt-4 max-w-2xl mx-auto sm:mx-0">
 | 
			
		||||
          Exploring <span class="font-medium text-zinc-900 dark:text-zinc-100">{sortedPosts.length}</span> articles tagged with <span class="font-medium text-zinc-900 dark:text-zinc-100">"{tag}"</span>
 | 
			
		||||
        <p
 | 
			
		||||
          class="mx-auto mt-4 max-w-2xl text-base text-zinc-600 dark:text-zinc-400 sm:mx-0 sm:text-lg"
 | 
			
		||||
        >
 | 
			
		||||
          Exploring <span class="font-medium text-zinc-900 dark:text-zinc-100"
 | 
			
		||||
            >{sortedPosts.length}</span
 | 
			
		||||
          > articles tagged with <span class="font-medium text-zinc-900 dark:text-zinc-100"
 | 
			
		||||
            >"{tag}"</span
 | 
			
		||||
          >
 | 
			
		||||
        </p>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Related tags section -->
 | 
			
		||||
    {relatedTags.length > 0 && (
 | 
			
		||||
      <div class="mb-8 sm:mb-12 overflow-x-auto pb-4 hide-scrollbar">
 | 
			
		||||
        <h2 class="text-lg font-medium text-zinc-900 dark:text-zinc-100 mb-3 text-center sm:text-left">Related topics</h2>
 | 
			
		||||
        <div class="flex gap-2 flex-nowrap justify-center sm:justify-start">
 | 
			
		||||
          {relatedTags.map(relatedTag => (
 | 
			
		||||
            <a
 | 
			
		||||
              href={`/topics/${relatedTag}`}
 | 
			
		||||
              class="flex-shrink-0 inline-flex items-center rounded-full px-3 py-1.5 text-sm font-medium bg-zinc-100 text-zinc-900 hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-100 dark:hover:bg-zinc-700 transition-colors"
 | 
			
		||||
            >
 | 
			
		||||
              #{relatedTag}
 | 
			
		||||
            </a>
 | 
			
		||||
          ))}
 | 
			
		||||
    {
 | 
			
		||||
      relatedTags.length > 0 && (
 | 
			
		||||
        <div class="hide-scrollbar mb-8 overflow-x-auto pb-4 sm:mb-12">
 | 
			
		||||
          <h2 class="mb-3 text-center text-lg font-medium text-zinc-900 dark:text-zinc-100 sm:text-left">
 | 
			
		||||
            Related topics
 | 
			
		||||
          </h2>
 | 
			
		||||
          <div class="flex flex-nowrap justify-center gap-2 sm:justify-start">
 | 
			
		||||
            {relatedTags.map((relatedTag) => (
 | 
			
		||||
              <a
 | 
			
		||||
                href={`/topics/${relatedTag}`}
 | 
			
		||||
                class="inline-flex flex-shrink-0 items-center rounded-full bg-zinc-100 px-3 py-1.5 text-sm font-medium text-zinc-900 transition-colors hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-100 dark:hover:bg-zinc-700"
 | 
			
		||||
              >
 | 
			
		||||
                #{relatedTag}
 | 
			
		||||
              </a>
 | 
			
		||||
            ))}
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    )}
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    <!-- Posts list -->
 | 
			
		||||
    <div class="relative">
 | 
			
		||||
      <div class="absolute inset-0 bg-grid-pattern opacity-5 dark:opacity-10 pointer-events-none"></div>
 | 
			
		||||
      <div class="bg-grid-pattern pointer-events-none absolute inset-0 opacity-5 dark:opacity-10">
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="relative space-y-6 sm:space-y-8">
 | 
			
		||||
        {sortedPosts.map((post) => (
 | 
			
		||||
          <article class="group relative flex flex-col p-5 sm:p-8 rounded-2xl border border-zinc-200 dark:border-zinc-800 hover:bg-zinc-50/80 dark:hover:bg-zinc-900/50 transition-all duration-300 hover:shadow-md hover-card max-w-2xl mx-auto sm:mx-0">
 | 
			
		||||
            <div class="absolute inset-0 bg-gradient-to-br from-zinc-50/0 to-zinc-100/0 dark:from-zinc-900/0 dark:to-zinc-800/0 opacity-0 group-hover:opacity-100 transition-opacity duration-500 rounded-2xl"></div>
 | 
			
		||||
        {
 | 
			
		||||
          sortedPosts.map((post) => (
 | 
			
		||||
            <article class="hover-card group relative mx-auto flex max-w-2xl flex-col rounded-2xl border border-zinc-200 p-5 transition-all duration-300 hover:bg-zinc-50/80 hover:shadow-md dark:border-zinc-800 dark:hover:bg-zinc-900/50 sm:mx-0 sm:p-8">
 | 
			
		||||
              <div class="absolute inset-0 rounded-2xl bg-gradient-to-br from-zinc-50/0 to-zinc-100/0 opacity-0 transition-opacity duration-500 group-hover:opacity-100 dark:from-zinc-900/0 dark:to-zinc-800/0" />
 | 
			
		||||
 | 
			
		||||
            <div class="flex flex-col sm:flex-row gap-5 sm:gap-6">
 | 
			
		||||
              {post.image && (
 | 
			
		||||
                <div class="flex-shrink-0 w-full sm:w-56 h-40 rounded-xl overflow-hidden shadow-sm group-hover:shadow-md transition-all duration-300 mx-auto sm:mx-0">
 | 
			
		||||
                  <img
 | 
			
		||||
                    src={`${process.env.DIRECTUS_URL ?? "https://directus.alexlebens.dev"}/assets/${post.image}?width=500`}
 | 
			
		||||
                    alt={post.image_alt}
 | 
			
		||||
                    class="w-full h-full object-cover transition-transform duration-500 group-hover:scale-105"
 | 
			
		||||
                    loading="lazy"
 | 
			
		||||
                  />
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
              <div class="flex flex-col gap-5 sm:flex-row sm:gap-6">
 | 
			
		||||
                {post.image && (
 | 
			
		||||
                  <div class="mx-auto h-40 w-full flex-shrink-0 overflow-hidden rounded-xl shadow-sm transition-all duration-300 group-hover:shadow-md sm:mx-0 sm:w-56">
 | 
			
		||||
                    <img
 | 
			
		||||
                      src={`${process.env.DIRECTUS_URL ?? 'https://directus.alexlebens.dev'}/assets/${post.image}?width=500`}
 | 
			
		||||
                      alt={post.image_alt}
 | 
			
		||||
                      class="h-full w-full object-cover transition-transform duration-500 group-hover:scale-105"
 | 
			
		||||
                      loading="lazy"
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                )}
 | 
			
		||||
 | 
			
		||||
              <div class="flex-1">
 | 
			
		||||
                <div class="flex flex-wrap items-center text-xs sm:text-sm text-zinc-500 dark:text-zinc-400 gap-3 sm:gap-4 mb-2 sm:mb-3 justify-center sm:justify-start">
 | 
			
		||||
                  {post.published_date && (
 | 
			
		||||
                    <time datetime={post.published_date.toLocaleString()} class="flex items-center gap-1.5">
 | 
			
		||||
                      <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-3.5 h-3.5 sm:w-4 sm:h-4">
 | 
			
		||||
                        <path stroke-linecap="round" stroke-linejoin="round" d="M6.75 3v2.25M17.25 3v2.25M3 18.75V7.5a2.25 2.25 0 012.25-2.25h13.5A2.25 2.25 0 0121 7.5v11.25m-18 0
 | 
			
		||||
                        A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75m-18 0v-7.5A2.25 2.25 0 015.25 9h13.5A2.25 2.25 0 0121 11.25v7.5" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                      <FormattedDate date={post.published_date} />
 | 
			
		||||
                    </time>
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="flex-1">
 | 
			
		||||
                  <div class="mb-2 flex flex-wrap items-center justify-center gap-3 text-xs text-zinc-500 dark:text-zinc-400 sm:mb-3 sm:justify-start sm:gap-4 sm:text-sm">
 | 
			
		||||
                    {post.published_date && (
 | 
			
		||||
                      <time
 | 
			
		||||
                        datetime={post.published_date.toLocaleString()}
 | 
			
		||||
                        class="flex items-center gap-1.5"
 | 
			
		||||
                      >
 | 
			
		||||
                        <svg
 | 
			
		||||
                          xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                          fill="none"
 | 
			
		||||
                          viewBox="0 0 24 24"
 | 
			
		||||
                          stroke-width="1.5"
 | 
			
		||||
                          stroke="currentColor"
 | 
			
		||||
                          class="h-3.5 w-3.5 sm:h-4 sm:w-4"
 | 
			
		||||
                        >
 | 
			
		||||
                          <path
 | 
			
		||||
                            stroke-linecap="round"
 | 
			
		||||
                            stroke-linejoin="round"
 | 
			
		||||
                            d="M6.75 3v2.25M17.25 3v2.25M3 18.75V7.5a2.25 2.25 0 012.25-2.25h13.5A2.25 2.25 0 0121 7.5v11.25m-18 0
 | 
			
		||||
                        A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75m-18 0v-7.5A2.25 2.25 0 015.25 9h13.5A2.25 2.25 0 0121 11.25v7.5"
 | 
			
		||||
                          />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                        <FormattedDate date={post.published_date} />
 | 
			
		||||
                      </time>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                <h2 class="text-xl sm:text-2xl font-semibold text-zinc-900 dark:text-zinc-100 mb-2 sm:mb-3 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors text-center sm:text-left">
 | 
			
		||||
                  <a href={`/blog/${post.slug}/`} class="before:absolute before:inset-0">
 | 
			
		||||
                    {post.title}
 | 
			
		||||
                  </a>
 | 
			
		||||
                </h2>
 | 
			
		||||
 | 
			
		||||
                <p class="text-sm sm:text-base text-zinc-600 dark:text-zinc-400 mb-4 line-clamp-2 sm:line-clamp-3 text-center sm:text-left">
 | 
			
		||||
                  {post.description}
 | 
			
		||||
                </p>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div class="flex flex-wrap justify-center sm:justify-between items-end mt-4 pt-4 border-t border-zinc-100 dark:border-zinc-800">
 | 
			
		||||
              {post.tags && post.tags.length > 0 && (
 | 
			
		||||
                <div class="flex flex-wrap gap-2 mb-3 sm:mb-0 justify-center sm:justify-start">
 | 
			
		||||
                  {post.tags.slice(0, 3).map(postTag => (
 | 
			
		||||
                    <a
 | 
			
		||||
                      href={`/topics/${postTag}`}
 | 
			
		||||
                      class={`inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium transition-colors ${
 | 
			
		||||
                        postTag === tag
 | 
			
		||||
                          ? 'bg-zinc-900/10 text-zinc-900 dark:bg-zinc-100/20 dark:text-zinc-100'
 | 
			
		||||
                          : 'bg-zinc-100 text-zinc-600 hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-400 dark:hover:bg-zinc-700'
 | 
			
		||||
                      }`}
 | 
			
		||||
                    >
 | 
			
		||||
                      #{postTag}
 | 
			
		||||
                  <h2 class="mb-2 text-center text-xl font-semibold text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-zinc-100 dark:group-hover:text-zinc-300 sm:mb-3 sm:text-left sm:text-2xl">
 | 
			
		||||
                    <a href={`/blog/${post.slug}/`} class="before:absolute before:inset-0">
 | 
			
		||||
                      {post.title}
 | 
			
		||||
                    </a>
 | 
			
		||||
                  ))}
 | 
			
		||||
                  {post.tags.length > 3 && (
 | 
			
		||||
                    <span class="inline-flex items-center rounded-full bg-zinc-50 px-2 py-0.5 text-xs text-zinc-500 dark:bg-zinc-800/50 dark:text-zinc-400">
 | 
			
		||||
                      +{post.tags.length - 3}
 | 
			
		||||
                    </span>
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
                  </h2>
 | 
			
		||||
 | 
			
		||||
              <div class="mx-auto sm:ml-auto sm:mr-0">
 | 
			
		||||
                <a
 | 
			
		||||
                  href={`/blog/${post.slug}/`}
 | 
			
		||||
                  class="inline-flex items-center text-sm font-medium text-zinc-700 dark:text-zinc-300 group-hover:text-zinc-900 dark:group-hover:text-zinc-100 transition-colors"
 | 
			
		||||
                  aria-hidden="true"
 | 
			
		||||
                  tabindex="-1"
 | 
			
		||||
                >
 | 
			
		||||
                  <span class="relative overflow-hidden inline-block">
 | 
			
		||||
                    <span class="block transition-transform duration-300 group-hover:-translate-y-full">Read article</span>
 | 
			
		||||
                    <span class="absolute top-0 left-0 translate-y-full group-hover:translate-y-0 transition-transform duration-300 whitespace-nowrap">Explore now</span>
 | 
			
		||||
                  </span>
 | 
			
		||||
                  <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 ml-1 transition-transform duration-300 group-hover:translate-x-1">
 | 
			
		||||
                    <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3" />
 | 
			
		||||
                  </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
                  <p class="mb-4 line-clamp-2 text-center text-sm text-zinc-600 dark:text-zinc-400 sm:line-clamp-3 sm:text-left sm:text-base">
 | 
			
		||||
                    {post.description}
 | 
			
		||||
                  </p>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </article>
 | 
			
		||||
        ))}
 | 
			
		||||
 | 
			
		||||
              <div class="mt-4 flex flex-wrap items-end justify-center border-t border-zinc-100 pt-4 dark:border-zinc-800 sm:justify-between">
 | 
			
		||||
                {post.tags && post.tags.length > 0 && (
 | 
			
		||||
                  <div class="mb-3 flex flex-wrap justify-center gap-2 sm:mb-0 sm:justify-start">
 | 
			
		||||
                    {post.tags.slice(0, 3).map((postTag) => (
 | 
			
		||||
                      <a
 | 
			
		||||
                        href={`/topics/${postTag}`}
 | 
			
		||||
                        class={`inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium transition-colors ${
 | 
			
		||||
                          postTag === tag
 | 
			
		||||
                            ? 'bg-zinc-900/10 text-zinc-900 dark:bg-zinc-100/20 dark:text-zinc-100'
 | 
			
		||||
                            : 'bg-zinc-100 text-zinc-600 hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-400 dark:hover:bg-zinc-700'
 | 
			
		||||
                        }`}
 | 
			
		||||
                      >
 | 
			
		||||
                        #{postTag}
 | 
			
		||||
                      </a>
 | 
			
		||||
                    ))}
 | 
			
		||||
                    {post.tags.length > 3 && (
 | 
			
		||||
                      <span class="inline-flex items-center rounded-full bg-zinc-50 px-2 py-0.5 text-xs text-zinc-500 dark:bg-zinc-800/50 dark:text-zinc-400">
 | 
			
		||||
                        +{post.tags.length - 3}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </div>
 | 
			
		||||
                )}
 | 
			
		||||
 | 
			
		||||
                <div class="mx-auto sm:ml-auto sm:mr-0">
 | 
			
		||||
                  <a
 | 
			
		||||
                    href={`/blog/${post.slug}/`}
 | 
			
		||||
                    class="inline-flex items-center text-sm font-medium text-zinc-700 transition-colors group-hover:text-zinc-900 dark:text-zinc-300 dark:group-hover:text-zinc-100"
 | 
			
		||||
                    aria-hidden="true"
 | 
			
		||||
                    tabindex="-1"
 | 
			
		||||
                  >
 | 
			
		||||
                    <span class="relative inline-block overflow-hidden">
 | 
			
		||||
                      <span class="block transition-transform duration-300 group-hover:-translate-y-full">
 | 
			
		||||
                        Read article
 | 
			
		||||
                      </span>
 | 
			
		||||
                      <span class="absolute left-0 top-0 translate-y-full whitespace-nowrap transition-transform duration-300 group-hover:translate-y-0">
 | 
			
		||||
                        Explore now
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <svg
 | 
			
		||||
                      xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                      fill="none"
 | 
			
		||||
                      viewBox="0 0 24 24"
 | 
			
		||||
                      stroke-width="1.5"
 | 
			
		||||
                      stroke="currentColor"
 | 
			
		||||
                      class="ml-1 h-4 w-4 transition-transform duration-300 group-hover:translate-x-1"
 | 
			
		||||
                    >
 | 
			
		||||
                      <path
 | 
			
		||||
                        stroke-linecap="round"
 | 
			
		||||
                        stroke-linejoin="round"
 | 
			
		||||
                        d="M13.5 4.5L21 12m0 0l-7.5 7.5M21 12H3"
 | 
			
		||||
                      />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                  </a>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </article>
 | 
			
		||||
          ))
 | 
			
		||||
        }
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <!-- Empty state với màu zinc -->
 | 
			
		||||
    {sortedPosts.length === 0 && (
 | 
			
		||||
      <div class="text-center py-12 sm:py-20">
 | 
			
		||||
        <div class="inline-flex items-center justify-center w-16 h-16 sm:w-20 sm:h-20 rounded-full bg-zinc-100 dark:bg-zinc-800 mb-4 sm:mb-6">
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-8 h-8 sm:w-10 sm:h-10 text-zinc-500 dark:text-zinc-400">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M12 9v3.75m9-.75a9 9 0 11-18 0 9 9 0 0118 0zm-9 3.75h.008v.008H12v-.008z" />
 | 
			
		||||
          </svg>
 | 
			
		||||
    {
 | 
			
		||||
      sortedPosts.length === 0 && (
 | 
			
		||||
        <div class="py-12 text-center sm:py-20">
 | 
			
		||||
          <div class="mb-4 inline-flex h-16 w-16 items-center justify-center rounded-full bg-zinc-100 dark:bg-zinc-800 sm:mb-6 sm:h-20 sm:w-20">
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="h-8 w-8 text-zinc-500 dark:text-zinc-400 sm:h-10 sm:w-10"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M12 9v3.75m9-.75a9 9 0 11-18 0 9 9 0 0118 0zm-9 3.75h.008v.008H12v-.008z"
 | 
			
		||||
              />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </div>
 | 
			
		||||
          <h2 class="mb-2 text-xl font-semibold text-zinc-900 dark:text-zinc-100 sm:text-2xl">
 | 
			
		||||
            No posts found
 | 
			
		||||
          </h2>
 | 
			
		||||
          <p class="text-zinc-600 dark:text-zinc-400">There are no posts with this tag yet.</p>
 | 
			
		||||
          <a
 | 
			
		||||
            href="/blog"
 | 
			
		||||
            class="mt-6 inline-flex items-center gap-2 rounded-md bg-zinc-100 px-4 py-2 text-sm font-medium text-zinc-800 transition-all duration-300 hover:bg-zinc-200 dark:bg-zinc-800 dark:text-zinc-200 dark:hover:bg-zinc-700"
 | 
			
		||||
          >
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="h-4 w-4"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M19.5 12h-15m0 0l6.75 6.75M4.5 12l6.75-6.75"
 | 
			
		||||
              />
 | 
			
		||||
            </svg>
 | 
			
		||||
            <span>Browse all articles</span>
 | 
			
		||||
          </a>
 | 
			
		||||
        </div>
 | 
			
		||||
        <h2 class="text-xl sm:text-2xl font-semibold text-zinc-900 dark:text-zinc-100 mb-2">No posts found</h2>
 | 
			
		||||
        <p class="text-zinc-600 dark:text-zinc-400">There are no posts with this tag yet.</p>
 | 
			
		||||
        <a href="/blog" class="inline-flex items-center gap-2 mt-6 px-4 py-2 rounded-md bg-zinc-100 dark:bg-zinc-800 text-zinc-800 dark:text-zinc-200 hover:bg-zinc-200 dark:hover:bg-zinc-700 transition-all duration-300 text-sm font-medium">
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M19.5 12h-15m0 0l6.75 6.75M4.5 12l6.75-6.75" />
 | 
			
		||||
          </svg>
 | 
			
		||||
          <span>Browse all articles</span>
 | 
			
		||||
        </a>
 | 
			
		||||
      </div>
 | 
			
		||||
    )}
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
  </div>
 | 
			
		||||
</BaseLayout>
 | 
			
		||||
 | 
			
		||||
@@ -237,8 +350,12 @@ const relatedTags = [...new Set(
 | 
			
		||||
 | 
			
		||||
  /* Animated underline */
 | 
			
		||||
  @keyframes expand {
 | 
			
		||||
    from { width: 0; }
 | 
			
		||||
    to { width: 50%; }
 | 
			
		||||
    from {
 | 
			
		||||
      width: 0;
 | 
			
		||||
    }
 | 
			
		||||
    to {
 | 
			
		||||
      width: 50%;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .animate-expand {
 | 
			
		||||
@@ -272,7 +389,10 @@ const relatedTags = [...new Set(
 | 
			
		||||
  /* Hover card effect */
 | 
			
		||||
  .hover-card {
 | 
			
		||||
    transform: translateY(0);
 | 
			
		||||
    transition: transform 0.3s ease, box-shadow 0.3s ease, background-color 0.3s ease;
 | 
			
		||||
    transition:
 | 
			
		||||
      transform 0.3s ease,
 | 
			
		||||
      box-shadow 0.3s ease,
 | 
			
		||||
      background-color 0.3s ease;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @media (hover: hover) {
 | 
			
		||||
@@ -308,11 +428,13 @@ const relatedTags = [...new Set(
 | 
			
		||||
  // Handle SPA transitions for tag pages
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all internal links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links, external links, or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') ||
 | 
			
		||||
          link.getAttribute('target') === '_blank' ||
 | 
			
		||||
          link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
      if (
 | 
			
		||||
        link.getAttribute('href').includes('#') ||
 | 
			
		||||
        link.getAttribute('target') === '_blank' ||
 | 
			
		||||
        link.hasAttribute('data-spa-handled')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
@@ -350,25 +472,34 @@ const relatedTags = [...new Set(
 | 
			
		||||
      // Animate header elements
 | 
			
		||||
      const headerElements = document.querySelectorAll('h1, .tag-icon, .tag-description');
 | 
			
		||||
      headerElements.forEach((el, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          el.classList.add('animate-reveal');
 | 
			
		||||
        }, 100 + (index * 150));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            el.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          100 + index * 150
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      // Animate posts with staggered delay
 | 
			
		||||
      const articles = document.querySelectorAll('article');
 | 
			
		||||
      articles.forEach((article, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          article.classList.add('animate-reveal');
 | 
			
		||||
        }, 400 + (index * 100));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            article.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          400 + index * 100
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      // Animate related tags
 | 
			
		||||
      const relatedTags = document.querySelectorAll('.related-tags a');
 | 
			
		||||
      relatedTags.forEach((tag, index) => {
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          tag.classList.add('animate-reveal');
 | 
			
		||||
        }, 600 + (index * 50));
 | 
			
		||||
        setTimeout(
 | 
			
		||||
          () => {
 | 
			
		||||
            tag.classList.add('animate-reveal');
 | 
			
		||||
          },
 | 
			
		||||
          600 + index * 50
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -387,4 +518,3 @@ const relatedTags = [...new Set(
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<!-- Add this at the end of your page -->
 | 
			
		||||
</BaseLayout>
 | 
			
		||||
@@ -1,110 +1,147 @@
 | 
			
		||||
---
 | 
			
		||||
import BaseLayout from '../../layouts/BaseLayout.astro';
 | 
			
		||||
 | 
			
		||||
import directus from "../../../lib/directus"
 | 
			
		||||
import { readItems } from "@directus/sdk";
 | 
			
		||||
import directus from '../../../lib/directus';
 | 
			
		||||
import { readItems } from '@directus/sdk';
 | 
			
		||||
 | 
			
		||||
const posts = await directus.request(
 | 
			
		||||
  readItems("posts", {
 | 
			
		||||
  readItems('posts', {
 | 
			
		||||
    fields: ['*'],
 | 
			
		||||
    sort: ["-published_date"],
 | 
			
		||||
    sort: ['-published_date'],
 | 
			
		||||
  })
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
const tags = [...new Set(posts.flatMap(post => post.tags || []))].sort();
 | 
			
		||||
const tags = [...new Set(posts.flatMap((post) => post.tags || []))].sort();
 | 
			
		||||
 | 
			
		||||
// Count posts for each tag and create tag objects with additional data
 | 
			
		||||
const tagObjects = tags.map(tag => {
 | 
			
		||||
  const count = posts.filter(post => post.tags?.includes(tag)).length;
 | 
			
		||||
const tagObjects = tags.map((tag) => {
 | 
			
		||||
  const count = posts.filter((post) => post.tags?.includes(tag)).length;
 | 
			
		||||
  // Generate a consistent but random-looking hue for each tag
 | 
			
		||||
  const hue = Math.abs(tag.split('').reduce((acc, char) => acc + char.charCodeAt(0), 0) % 360);
 | 
			
		||||
  return { 
 | 
			
		||||
    name: tag, 
 | 
			
		||||
  return {
 | 
			
		||||
    name: tag,
 | 
			
		||||
    count,
 | 
			
		||||
    size: Math.max(1, Math.min(3, Math.floor(count / 2) + 1)), // Size 1-3 based on count
 | 
			
		||||
    hue
 | 
			
		||||
    hue,
 | 
			
		||||
  };
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
<BaseLayout title="Explore Tags">
 | 
			
		||||
  <div class="w-full mx-auto px-3 sm:px-6 py-6 sm:py-12 md:py-16 theme-transition-all">
 | 
			
		||||
  <div class="theme-transition-all mx-auto w-full px-3 py-6 sm:px-6 sm:py-12 md:py-16">
 | 
			
		||||
    <!-- Enhanced header section with animated elements - improved for mobile -->
 | 
			
		||||
    <div class="relative mb-8 sm:mb-12 md:mb-16 text-center theme-transition-element">
 | 
			
		||||
      <div class="absolute -top-16 -left-16 w-36 sm:w-48 md:w-72 h-36 sm:h-48 md:h-72 bg-zinc-100 dark:bg-zinc-800/50 rounded-full blur-3xl opacity-50 animate-blob theme-transition-bg"></div>
 | 
			
		||||
      <div class="absolute -bottom-16 -right-16 w-36 sm:w-48 md:w-72 h-36 sm:h-48 md:h-72 bg-zinc-200 dark:bg-zinc-800/30 rounded-full blur-3xl opacity-30 animate-blob animation-delay-2000 theme-transition-bg"></div>
 | 
			
		||||
      <div class="absolute top-8 right-8 w-24 sm:w-32 md:w-40 h-24 sm:h-32 md:h-40 bg-zinc-100/30 dark:bg-zinc-700/20 rounded-full blur-2xl opacity-40 animate-blob animation-delay-4000 theme-transition-bg"></div>
 | 
			
		||||
      
 | 
			
		||||
      <h1 class="relative text-3xl sm:text-4xl md:text-5xl lg:text-6xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 mb-3 sm:mb-4 md:mb-6 theme-transition-color">
 | 
			
		||||
        <span class="inline-block relative">
 | 
			
		||||
    <div class="theme-transition-element relative mb-8 text-center sm:mb-12 md:mb-16">
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob theme-transition-bg absolute -left-16 -top-16 h-36 w-36 rounded-full bg-zinc-100 opacity-50 blur-3xl dark:bg-zinc-800/50 sm:h-48 sm:w-48 md:h-72 md:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-2000 theme-transition-bg absolute -bottom-16 -right-16 h-36 w-36 rounded-full bg-zinc-200 opacity-30 blur-3xl dark:bg-zinc-800/30 sm:h-48 sm:w-48 md:h-72 md:w-72"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="animate-blob animation-delay-4000 theme-transition-bg absolute right-8 top-8 h-24 w-24 rounded-full bg-zinc-100/30 opacity-40 blur-2xl dark:bg-zinc-700/20 sm:h-32 sm:w-32 md:h-40 md:w-40"
 | 
			
		||||
      >
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <h1
 | 
			
		||||
        class="theme-transition-color relative mb-3 text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100 sm:mb-4 sm:text-4xl md:mb-6 md:text-5xl lg:text-6xl"
 | 
			
		||||
      >
 | 
			
		||||
        <span class="relative inline-block">
 | 
			
		||||
          <span class="relative inline-block">
 | 
			
		||||
            <span class="absolute -inset-1 rounded-lg bg-gradient-to-r from-zinc-200/50 to-zinc-300/50 dark:from-zinc-800/50 dark:to-zinc-700/50 blur-sm theme-transition-bg"></span>
 | 
			
		||||
            <span
 | 
			
		||||
              class="theme-transition-bg absolute -inset-1 rounded-lg bg-gradient-to-r from-zinc-200/50 to-zinc-300/50 blur-sm dark:from-zinc-800/50 dark:to-zinc-700/50"
 | 
			
		||||
            ></span>
 | 
			
		||||
            <span class="relative">Explore</span>
 | 
			
		||||
          </span>
 | 
			
		||||
          {" "}
 | 
			
		||||
          {' '}
 | 
			
		||||
          <span class="relative inline-block">
 | 
			
		||||
            Topics
 | 
			
		||||
            <span class="absolute -bottom-1 sm:-bottom-2 left-0 w-full h-0.5 sm:h-1 bg-gradient-to-r from-zinc-400 to-zinc-600 dark:from-zinc-600 dark:to-zinc-400 transform origin-left animate-underline theme-transition-bg"></span>
 | 
			
		||||
            <span
 | 
			
		||||
              class="animate-underline theme-transition-bg absolute -bottom-1 left-0 h-0.5 w-full origin-left transform bg-gradient-to-r from-zinc-400 to-zinc-600 dark:from-zinc-600 dark:to-zinc-400 sm:-bottom-2 sm:h-1"
 | 
			
		||||
            ></span>
 | 
			
		||||
          </span>
 | 
			
		||||
        </span>
 | 
			
		||||
      </h1>
 | 
			
		||||
      <p class="relative text-sm sm:text-base md:text-lg lg:text-xl text-zinc-600 dark:text-zinc-400 max-w-2xl mx-auto theme-transition-color">
 | 
			
		||||
      <p
 | 
			
		||||
        class="theme-transition-color relative mx-auto max-w-2xl text-sm text-zinc-600 dark:text-zinc-400 sm:text-base md:text-lg lg:text-xl"
 | 
			
		||||
      >
 | 
			
		||||
        Discover content organized by your interests
 | 
			
		||||
      </p>
 | 
			
		||||
    </div>
 | 
			
		||||
    
 | 
			
		||||
    {tags.length === 0 ? (
 | 
			
		||||
      <div class="text-center py-8 sm:py-12 md:py-16 theme-transition-element">
 | 
			
		||||
        <div class="inline-flex items-center justify-center w-16 sm:w-20 md:w-24 h-16 sm:h-20 md:h-24 rounded-full bg-zinc-100 dark:bg-zinc-800 mb-3 sm:mb-4 md:mb-6 shadow-inner theme-transition-bg">
 | 
			
		||||
          <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-8 sm:w-10 md:w-12 h-8 sm:h-10 md:h-12 text-zinc-500 dark:text-zinc-400 theme-transition-color">
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M9.568 3H5.25A2.25 2.25 0 003 5.25v4.318c0 .597.237 1.17.659 1.591l9.581 9.581c.699.699 1.78.872 2.607.33a18.095 18.095 0 005.223-5.223c.542-.827.369-1.908-.33-2.607L11.16 3.66A2.25 2.25 0 009.568 3z" />
 | 
			
		||||
            <path stroke-linecap="round" stroke-linejoin="round" d="M6 6h.008v.008H6V6z" />
 | 
			
		||||
          </svg>
 | 
			
		||||
 | 
			
		||||
    {
 | 
			
		||||
      tags.length === 0 ? (
 | 
			
		||||
        <div class="theme-transition-element py-8 text-center sm:py-12 md:py-16">
 | 
			
		||||
          <div class="theme-transition-bg mb-3 inline-flex h-16 w-16 items-center justify-center rounded-full bg-zinc-100 shadow-inner dark:bg-zinc-800 sm:mb-4 sm:h-20 sm:w-20 md:mb-6 md:h-24 md:w-24">
 | 
			
		||||
            <svg
 | 
			
		||||
              xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              fill="none"
 | 
			
		||||
              viewBox="0 0 24 24"
 | 
			
		||||
              stroke-width="1.5"
 | 
			
		||||
              stroke="currentColor"
 | 
			
		||||
              class="theme-transition-color h-8 w-8 text-zinc-500 dark:text-zinc-400 sm:h-10 sm:w-10 md:h-12 md:w-12"
 | 
			
		||||
            >
 | 
			
		||||
              <path
 | 
			
		||||
                stroke-linecap="round"
 | 
			
		||||
                stroke-linejoin="round"
 | 
			
		||||
                d="M9.568 3H5.25A2.25 2.25 0 003 5.25v4.318c0 .597.237 1.17.659 1.591l9.581 9.581c.699.699 1.78.872 2.607.33a18.095 18.095 0 005.223-5.223c.542-.827.369-1.908-.33-2.607L11.16 3.66A2.25 2.25 0 009.568 3z"
 | 
			
		||||
              />
 | 
			
		||||
              <path stroke-linecap="round" stroke-linejoin="round" d="M6 6h.008v.008H6V6z" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </div>
 | 
			
		||||
          <p class="theme-transition-color text-lg font-medium text-zinc-800 dark:text-zinc-200 sm:text-xl md:text-2xl">
 | 
			
		||||
            No tags found yet.
 | 
			
		||||
          </p>
 | 
			
		||||
          <p class="theme-transition-color mt-2 text-xs text-zinc-500 dark:text-zinc-500 sm:text-sm md:text-base">
 | 
			
		||||
            Check back later for categorized content.
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
        <p class="text-lg sm:text-xl md:text-2xl font-medium text-zinc-800 dark:text-zinc-200 theme-transition-color">No tags found yet.</p>
 | 
			
		||||
        <p class="mt-2 text-xs sm:text-sm md:text-base text-zinc-500 dark:text-zinc-500 theme-transition-color">Check back later for categorized content.</p>
 | 
			
		||||
      </div>
 | 
			
		||||
    ) : (
 | 
			
		||||
      <div class="flex justify-center w-full">
 | 
			
		||||
        <!-- Featured Tags Section - ultra-responsive design -->
 | 
			
		||||
        <div class="tag-cloud relative p-3 sm:p-4 md:p-6 lg:p-8 rounded-lg sm:rounded-xl md:rounded-2xl lg:rounded-3xl border border-zinc-100 dark:border-zinc-800 bg-white/50 dark:bg-zinc-900/50 backdrop-blur-sm hover-3d glass theme-transition-all w-full">
 | 
			
		||||
          <div class="absolute inset-0 bg-grid-pattern opacity-5 dark:opacity-10 theme-transition-bg"></div>
 | 
			
		||||
          <div class="absolute -top-8 -right-8 w-20 sm:w-24 md:w-32 lg:w-40 h-20 sm:h-24 md:h-32 lg:h-40 bg-gradient-to-br from-zinc-200/30 to-zinc-300/20 dark:from-zinc-700/20 dark:to-zinc-800/10 rounded-full blur-xl theme-transition-bg"></div>
 | 
			
		||||
          <div class="absolute -bottom-8 -left-8 w-20 sm:w-24 md:w-32 lg:w-40 h-20 sm:h-24 md:h-32 lg:h-40 bg-gradient-to-tl from-zinc-200/30 to-zinc-300/20 dark:from-zinc-700/20 dark:to-zinc-800/10 rounded-full blur-xl theme-transition-bg"></div>
 | 
			
		||||
          
 | 
			
		||||
          <h2 class="text-lg sm:text-xl md:text-2xl lg:text-3xl font-bold text-zinc-900 dark:text-zinc-100 mb-3 sm:mb-4 md:mb-6 lg:mb-8 text-center theme-transition-color">Popular Topics</h2>
 | 
			
		||||
          
 | 
			
		||||
          <!-- Ultra-responsive grid layout with fallbacks -->
 | 
			
		||||
          <div class="grid grid-cols-2 xxxs:grid-cols-2 xxs:grid-cols-2 xs:grid-cols-3 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-1.5 xxxs:gap-2 xxs:gap-2 xs:gap-2 sm:gap-3 md:gap-4 w-full">
 | 
			
		||||
            {sortedTags.map((tag) => (
 | 
			
		||||
              <a 
 | 
			
		||||
                href={`/topics/${tag.name}`}
 | 
			
		||||
                class="group relative overflow-hidden rounded-md sm:rounded-lg md:rounded-xl border border-zinc-200 dark:border-zinc-800 transition-all duration-300 hover:shadow-md sm:hover:shadow-lg hover:scale-[1.03] hover:border-zinc-300 dark:hover:border-zinc-700 active:scale-95 theme-transition-element theme-ripple flex-grow min-w-0"
 | 
			
		||||
                style={`--tag-hue: ${tag.hue};`}
 | 
			
		||||
              >
 | 
			
		||||
                <div class="absolute inset-0 bg-gradient-to-br from-zinc-50/90 to-zinc-100/90 dark:from-zinc-800/90 dark:to-zinc-900/90 opacity-100 group-hover:opacity-95 transition-opacity theme-transition-bg"></div>
 | 
			
		||||
                
 | 
			
		||||
                <div class="relative px-1.5 xxxs:px-2 xxs:px-2 xs:px-2 sm:px-3 md:px-4 py-1.5 xxxs:py-2 xxs:py-2 xs:py-2 sm:py-3 md:py-4 flex items-center gap-1.5 xxs:gap-2 w-full">
 | 
			
		||||
                  <div class="flex-shrink-0 flex items-center justify-center w-5 h-5 xxxs:w-6 xxxs:h-6 xxs:w-6 xxs:h-6 xs:w-7 xs:h-7 sm:w-8 sm:h-8 md:w-10 md:h-10 rounded-full bg-zinc-100 dark:bg-zinc-800 text-zinc-700 dark:text-zinc-300 group-hover:bg-accent/20 dark:group-hover:bg-accent/20 group-hover:text-accent-dark dark:group-hover:text-accent-light transition-all duration-300 shadow-sm theme-transition-all">
 | 
			
		||||
                    <span class="text-xs xxxs:text-xs xxs:text-xs xs:text-sm sm:text-base md:text-lg font-semibold">#</span>
 | 
			
		||||
      ) : (
 | 
			
		||||
        <div class="flex w-full justify-center">
 | 
			
		||||
          <div class="tag-cloud hover-3d glass theme-transition-all relative w-full rounded-lg border border-zinc-100 bg-white/50 p-3 backdrop-blur-sm dark:border-zinc-800 dark:bg-zinc-900/50 sm:rounded-xl sm:p-4 md:rounded-2xl md:p-6 lg:rounded-3xl lg:p-8">
 | 
			
		||||
            <div class="bg-grid-pattern theme-transition-bg absolute inset-0 opacity-5 dark:opacity-10" />
 | 
			
		||||
            <div class="theme-transition-bg absolute -right-8 -top-8 h-20 w-20 rounded-full bg-gradient-to-br from-zinc-200/30 to-zinc-300/20 blur-xl dark:from-zinc-700/20 dark:to-zinc-800/10 sm:h-24 sm:w-24 md:h-32 md:w-32 lg:h-40 lg:w-40" />
 | 
			
		||||
            <div class="theme-transition-bg absolute -bottom-8 -left-8 h-20 w-20 rounded-full bg-gradient-to-tl from-zinc-200/30 to-zinc-300/20 blur-xl dark:from-zinc-700/20 dark:to-zinc-800/10 sm:h-24 sm:w-24 md:h-32 md:w-32 lg:h-40 lg:w-40" />
 | 
			
		||||
 | 
			
		||||
            <h2 class="theme-transition-color mb-3 text-center text-lg font-bold text-zinc-900 dark:text-zinc-100 sm:mb-4 sm:text-xl md:mb-6 md:text-2xl lg:mb-8 lg:text-3xl">
 | 
			
		||||
              Popular Topics
 | 
			
		||||
            </h2>
 | 
			
		||||
 | 
			
		||||
            <div class="xxxs:grid-cols-2 xxs:grid-cols-2 xs:grid-cols-3 xxxs:gap-2 xxs:gap-2 xs:gap-2 grid w-full grid-cols-2 gap-1.5 sm:grid-cols-3 sm:gap-3 md:grid-cols-4 md:gap-4 lg:grid-cols-5">
 | 
			
		||||
              {sortedTags.map((tag) => (
 | 
			
		||||
                <a
 | 
			
		||||
                  href={`/topics/${tag.name}`}
 | 
			
		||||
                  class="theme-transition-element theme-ripple group relative min-w-0 flex-grow overflow-hidden rounded-md border border-zinc-200 transition-all duration-300 hover:scale-[1.03] hover:border-zinc-300 hover:shadow-md active:scale-95 dark:border-zinc-800 dark:hover:border-zinc-700 sm:rounded-lg sm:hover:shadow-lg md:rounded-xl"
 | 
			
		||||
                  style={`--tag-hue: ${tag.hue};`}
 | 
			
		||||
                >
 | 
			
		||||
                  <div class="theme-transition-bg absolute inset-0 bg-gradient-to-br from-zinc-50/90 to-zinc-100/90 opacity-100 transition-opacity group-hover:opacity-95 dark:from-zinc-800/90 dark:to-zinc-900/90" />
 | 
			
		||||
 | 
			
		||||
                  <div class="xxxs:px-2 xxs:px-2 xs:px-2 xxxs:py-2 xxs:py-2 xs:py-2 xxs:gap-2 relative flex w-full items-center gap-1.5 px-1.5 py-1.5 sm:px-3 sm:py-3 md:px-4 md:py-4">
 | 
			
		||||
                    <div class="xxxs:w-6 xxxs:h-6 xxs:w-6 xxs:h-6 xs:w-7 xs:h-7 group-hover:bg-accent/20 dark:group-hover:bg-accent/20 group-hover:text-accent-dark dark:group-hover:text-accent-light theme-transition-all flex h-5 w-5 flex-shrink-0 items-center justify-center rounded-full bg-zinc-100 text-zinc-700 shadow-sm transition-all duration-300 dark:bg-zinc-800 dark:text-zinc-300 sm:h-8 sm:w-8 md:h-10 md:w-10">
 | 
			
		||||
                      <span class="xxxs:text-xs xxs:text-xs xs:text-sm text-xs font-semibold sm:text-base md:text-lg">
 | 
			
		||||
                        #
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
 | 
			
		||||
                    <div class="min-w-0 flex-1 overflow-hidden">
 | 
			
		||||
                      <h3 class="xxxs:text-xs xxs:text-xs xs:text-xs theme-transition-color truncate hyphens-auto break-words text-[10px] font-bold text-zinc-900 transition-colors group-hover:text-zinc-700 dark:text-zinc-100 dark:group-hover:text-zinc-300 sm:text-sm md:text-base">
 | 
			
		||||
                        {tag.name}
 | 
			
		||||
                      </h3>
 | 
			
		||||
                      <p class="xxxs:text-[9px] xxs:text-[9px] xs:text-[10px] theme-transition-color truncate text-[8px] text-zinc-500 dark:text-zinc-400 sm:text-xs md:text-xs">
 | 
			
		||||
                        {tag.count} article{tag.count !== 1 ? 's' : ''}
 | 
			
		||||
                      </p>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  
 | 
			
		||||
                  <div class="flex-1 min-w-0 overflow-hidden">
 | 
			
		||||
                    <h3 class="text-[10px] xxxs:text-xs xxs:text-xs xs:text-xs sm:text-sm md:text-base font-bold text-zinc-900 dark:text-zinc-100 group-hover:text-zinc-700 dark:group-hover:text-zinc-300 transition-colors theme-transition-color break-words hyphens-auto truncate">
 | 
			
		||||
                      {tag.name}
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    <p class="text-[8px] xxxs:text-[9px] xxs:text-[9px] xs:text-[10px] sm:text-xs md:text-xs text-zinc-500 dark:text-zinc-400 theme-transition-color truncate">{tag.count} article{tag.count !== 1 ? 's' : ''}</p>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </a>
 | 
			
		||||
            ))}
 | 
			
		||||
                </a>
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    )}
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
  </div>
 | 
			
		||||
</BaseLayout>
 | 
			
		||||
 | 
			
		||||
@@ -121,46 +158,50 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        meta.content = 'width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no';
 | 
			
		||||
        document.getElementsByTagName('head')[0].appendChild(meta);
 | 
			
		||||
      } else {
 | 
			
		||||
        viewport.setAttribute('content', 'width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no');
 | 
			
		||||
        viewport.setAttribute(
 | 
			
		||||
          'content',
 | 
			
		||||
          'width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no'
 | 
			
		||||
        );
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Fix for horizontal overflow
 | 
			
		||||
      document.body.style.overflowX = 'hidden';
 | 
			
		||||
      document.documentElement.style.overflowX = 'hidden';
 | 
			
		||||
      document.documentElement.style.width = '100%';
 | 
			
		||||
      document.body.style.width = '100%';
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    fixViewportWidth();
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Adjust tag items based on screen size with extreme precision
 | 
			
		||||
    const adjustTagItems = () => {
 | 
			
		||||
      const tagItems = document.querySelectorAll('.theme-ripple');
 | 
			
		||||
      const width = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;
 | 
			
		||||
      const width =
 | 
			
		||||
        window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;
 | 
			
		||||
      const isVerySmall = width < 360;
 | 
			
		||||
      const isExtremelySmall = width < 280;
 | 
			
		||||
      const isMicroScreen = width < 240;
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Fix container width to match viewport exactly
 | 
			
		||||
      const container = document.querySelector('.tag-cloud');
 | 
			
		||||
      if (container) {
 | 
			
		||||
        container.style.maxWidth = '100vw';
 | 
			
		||||
        container.style.width = '100%';
 | 
			
		||||
        container.style.boxSizing = 'border-box';
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Remove any margins that might cause overflow
 | 
			
		||||
        container.style.marginLeft = '0';
 | 
			
		||||
        container.style.marginRight = '0';
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Fix grid width
 | 
			
		||||
      const grid = document.querySelector('.grid');
 | 
			
		||||
      if (grid) {
 | 
			
		||||
        grid.style.width = '100%';
 | 
			
		||||
        grid.style.maxWidth = '100%';
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
      tagItems.forEach(item => {
 | 
			
		||||
 | 
			
		||||
      tagItems.forEach((item) => {
 | 
			
		||||
        // Set appropriate classes based on screen size
 | 
			
		||||
        if (isMicroScreen) {
 | 
			
		||||
          item.classList.add('micro-screen');
 | 
			
		||||
@@ -174,15 +215,15 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        } else {
 | 
			
		||||
          item.classList.remove('very-small-screen', 'extremely-small-screen', 'micro-screen');
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Ensure text doesn't overflow on small screens
 | 
			
		||||
        const tagName = item.querySelector('h3');
 | 
			
		||||
        const tagCount = item.querySelector('p');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        if (tagName) {
 | 
			
		||||
          // Set title for accessibility
 | 
			
		||||
          tagName.title = tagName.textContent.trim();
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Adjust text length based on screen size
 | 
			
		||||
          if (isMicroScreen && tagName.textContent.length > 6) {
 | 
			
		||||
            tagName.dataset.fullText = tagName.textContent;
 | 
			
		||||
@@ -198,32 +239,32 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
            delete tagName.dataset.fullText;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Set the tag hue for hover effects
 | 
			
		||||
        const hue = item.style.getPropertyValue('--tag-hue');
 | 
			
		||||
        item.style.setProperty('--hover-hue', hue);
 | 
			
		||||
      });
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Run on load
 | 
			
		||||
    adjustTagItems();
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Run on resize with optimized debounce
 | 
			
		||||
    let resizeTimer;
 | 
			
		||||
    const handleResize = () => {
 | 
			
		||||
      if (resizeTimer) {
 | 
			
		||||
        window.cancelAnimationFrame(resizeTimer);
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      resizeTimer = window.requestAnimationFrame(() => {
 | 
			
		||||
        fixViewportWidth();
 | 
			
		||||
        adjustTagItems();
 | 
			
		||||
      });
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    window.addEventListener('resize', handleResize);
 | 
			
		||||
    window.addEventListener('orientationchange', handleResize);
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Ensure layout is recalculated after page is fully loaded
 | 
			
		||||
    window.addEventListener('load', () => {
 | 
			
		||||
      fixViewportWidth();
 | 
			
		||||
@@ -234,17 +275,20 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        adjustTagItems();
 | 
			
		||||
      }, 500);
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Fix for iOS Safari and other mobile browsers
 | 
			
		||||
    if (/iPhone|iPad|iPod|Android/.test(navigator.userAgent)) {
 | 
			
		||||
      document.documentElement.style.setProperty('--safe-area-inset-bottom', 'env(safe-area-inset-bottom)');
 | 
			
		||||
      
 | 
			
		||||
      document.documentElement.style.setProperty(
 | 
			
		||||
        '--safe-area-inset-bottom',
 | 
			
		||||
        'env(safe-area-inset-bottom)'
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      // Fix for mobile viewport height issues
 | 
			
		||||
      const setVh = () => {
 | 
			
		||||
        const vh = window.innerHeight * 0.01;
 | 
			
		||||
        document.documentElement.style.setProperty('--vh', `${vh}px`);
 | 
			
		||||
      };
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      setVh();
 | 
			
		||||
      window.addEventListener('resize', setVh);
 | 
			
		||||
      window.addEventListener('orientationchange', () => {
 | 
			
		||||
@@ -255,39 +299,51 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        }, 100);
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Add touch support for mobile devices
 | 
			
		||||
    const addTouchSupport = () => {
 | 
			
		||||
      const tagItems = document.querySelectorAll('.theme-ripple');
 | 
			
		||||
      
 | 
			
		||||
      tagItems.forEach(item => {
 | 
			
		||||
        item.addEventListener('touchstart', () => {
 | 
			
		||||
          item.classList.add('touch-active');
 | 
			
		||||
        }, { passive: true });
 | 
			
		||||
        
 | 
			
		||||
        item.addEventListener('touchend', () => {
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            item.classList.remove('touch-active');
 | 
			
		||||
          }, 150);
 | 
			
		||||
        }, { passive: true });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
      tagItems.forEach((item) => {
 | 
			
		||||
        item.addEventListener(
 | 
			
		||||
          'touchstart',
 | 
			
		||||
          () => {
 | 
			
		||||
            item.classList.add('touch-active');
 | 
			
		||||
          },
 | 
			
		||||
          { passive: true }
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        item.addEventListener(
 | 
			
		||||
          'touchend',
 | 
			
		||||
          () => {
 | 
			
		||||
            setTimeout(() => {
 | 
			
		||||
              item.classList.remove('touch-active');
 | 
			
		||||
            }, 150);
 | 
			
		||||
          },
 | 
			
		||||
          { passive: true }
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        // Cancel active state if touch moves away
 | 
			
		||||
        item.addEventListener('touchmove', (e) => {
 | 
			
		||||
          const touch = e.touches[0];
 | 
			
		||||
          const rect = item.getBoundingClientRect();
 | 
			
		||||
          
 | 
			
		||||
          if (
 | 
			
		||||
            touch.clientX < rect.left || 
 | 
			
		||||
            touch.clientX > rect.right || 
 | 
			
		||||
            touch.clientY < rect.top || 
 | 
			
		||||
            touch.clientY > rect.bottom
 | 
			
		||||
          ) {
 | 
			
		||||
            item.classList.remove('touch-active');
 | 
			
		||||
          }
 | 
			
		||||
        }, { passive: true });
 | 
			
		||||
        item.addEventListener(
 | 
			
		||||
          'touchmove',
 | 
			
		||||
          (e) => {
 | 
			
		||||
            const touch = e.touches[0];
 | 
			
		||||
            const rect = item.getBoundingClientRect();
 | 
			
		||||
 | 
			
		||||
            if (
 | 
			
		||||
              touch.clientX < rect.left ||
 | 
			
		||||
              touch.clientX > rect.right ||
 | 
			
		||||
              touch.clientY < rect.top ||
 | 
			
		||||
              touch.clientY > rect.bottom
 | 
			
		||||
            ) {
 | 
			
		||||
              item.classList.remove('touch-active');
 | 
			
		||||
            }
 | 
			
		||||
          },
 | 
			
		||||
          { passive: true }
 | 
			
		||||
        );
 | 
			
		||||
      });
 | 
			
		||||
    };
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    addTouchSupport();
 | 
			
		||||
  });
 | 
			
		||||
</script>
 | 
			
		||||
@@ -295,9 +351,10 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
<style>
 | 
			
		||||
  /* Base styles */
 | 
			
		||||
  .tag-cloud {
 | 
			
		||||
    box-shadow: 0 0 0 1px rgba(0, 0, 0, 0.03), 
 | 
			
		||||
                0 2px 4px rgba(0, 0, 0, 0.03), 
 | 
			
		||||
                0 4px 8px rgba(0, 0, 0, 0.05);
 | 
			
		||||
    box-shadow:
 | 
			
		||||
      0 0 0 1px rgba(0, 0, 0, 0.03),
 | 
			
		||||
      0 2px 4px rgba(0, 0, 0, 0.03),
 | 
			
		||||
      0 4px 8px rgba(0, 0, 0, 0.05);
 | 
			
		||||
    transform-style: preserve-3d;
 | 
			
		||||
    perspective: 1000px;
 | 
			
		||||
    transition: all var(--theme-transition);
 | 
			
		||||
@@ -307,19 +364,20 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
    margin-left: 0 !important;
 | 
			
		||||
    margin-right: 0 !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Fix for horizontal overflow */
 | 
			
		||||
  :global(html), :global(body) {
 | 
			
		||||
  :global(html),
 | 
			
		||||
  :global(body) {
 | 
			
		||||
    overflow-x: hidden;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    max-width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  :global(.max-w-6xl) {
 | 
			
		||||
    max-width: 100% !important;
 | 
			
		||||
    width: 100% !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Ultra-responsive breakpoints for extreme reliability */
 | 
			
		||||
  /* Micro screens (below 240px) */
 | 
			
		||||
  @media (max-width: 239px) {
 | 
			
		||||
@@ -329,146 +387,146 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
      border-radius: 0.25rem !important;
 | 
			
		||||
      width: 100% !important;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .tag-cloud h2 {
 | 
			
		||||
      font-size: 0.875rem !important;
 | 
			
		||||
      margin-bottom: 0.375rem !important;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .grid {
 | 
			
		||||
      grid-template-columns: repeat(1, minmax(0, 1fr)) !important;
 | 
			
		||||
      gap: 0.375rem !important;
 | 
			
		||||
      width: 100% !important;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .micro-screen .flex {
 | 
			
		||||
      padding: 0.25rem !important;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .micro-screen h3 {
 | 
			
		||||
      font-size: 0.625rem !important;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .micro-screen p {
 | 
			
		||||
      font-size: 0.5rem !important;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Extra extra extra small screens (240px-279px) */
 | 
			
		||||
  @media (min-width: 240px) and (max-width: 279px) {
 | 
			
		||||
    .xxxs\:grid-cols-2 {
 | 
			
		||||
      grid-template-columns: repeat(2, minmax(0, 1fr));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:px-2 {
 | 
			
		||||
      padding-left: 0.5rem;
 | 
			
		||||
      padding-right: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:py-2 {
 | 
			
		||||
      padding-top: 0.5rem;
 | 
			
		||||
      padding-bottom: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:w-6 {
 | 
			
		||||
      width: 1.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:h-6 {
 | 
			
		||||
      height: 1.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:text-xs {
 | 
			
		||||
      font-size: 0.75rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:gap-2 {
 | 
			
		||||
      gap: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxxs\:text-\[9px\] {
 | 
			
		||||
      font-size: 9px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Extra extra small screens (280px-359px) */
 | 
			
		||||
  @media (min-width: 280px) {
 | 
			
		||||
    .xxs\:grid-cols-2 {
 | 
			
		||||
      grid-template-columns: repeat(2, minmax(0, 1fr));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:px-2 {
 | 
			
		||||
      padding-left: 0.5rem;
 | 
			
		||||
      padding-right: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:py-2 {
 | 
			
		||||
      padding-top: 0.5rem;
 | 
			
		||||
      padding-bottom: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:w-6 {
 | 
			
		||||
      width: 1.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:h-6 {
 | 
			
		||||
      height: 1.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:text-xs {
 | 
			
		||||
      font-size: 0.75rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:gap-2 {
 | 
			
		||||
      gap: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xxs\:text-\[9px\] {
 | 
			
		||||
      font-size: 9px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Extra small screens (360px-639px) */
 | 
			
		||||
  @media (min-width: 360px) {
 | 
			
		||||
    .xs\:grid-cols-3 {
 | 
			
		||||
      grid-template-columns: repeat(3, minmax(0, 1fr));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:px-2 {
 | 
			
		||||
      padding-left: 0.5rem;
 | 
			
		||||
      padding-right: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:py-2 {
 | 
			
		||||
      padding-top: 0.5rem;
 | 
			
		||||
      padding-bottom: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:w-7 {
 | 
			
		||||
      width: 1.75rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:h-7 {
 | 
			
		||||
      height: 1.75rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:text-xs {
 | 
			
		||||
      font-size: 0.75rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:text-sm {
 | 
			
		||||
      font-size: 0.875rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:gap-2 {
 | 
			
		||||
      gap: 0.5rem;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    .xs\:text-\[10px\] {
 | 
			
		||||
      font-size: 10px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Ensure text doesn't overflow on small screens */
 | 
			
		||||
  .truncate {
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
@@ -476,7 +534,7 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
    white-space: nowrap;
 | 
			
		||||
    max-width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Ensure proper word breaking for long tag names */
 | 
			
		||||
  .break-words {
 | 
			
		||||
    word-break: break-word;
 | 
			
		||||
@@ -486,38 +544,42 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
  .hyphens-auto {
 | 
			
		||||
    hyphens: auto;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Improved shadow for dark mode */
 | 
			
		||||
  :global(.dark) .tag-cloud {
 | 
			
		||||
    box-shadow: 0 0 0 1px rgba(255, 255, 255, 0.05),
 | 
			
		||||
                0 2px 4px rgba(0, 0, 0, 0.1),
 | 
			
		||||
                0 4px 8px rgba(0, 0, 0, 0.15);
 | 
			
		||||
    box-shadow:
 | 
			
		||||
      0 0 0 1px rgba(255, 255, 255, 0.05),
 | 
			
		||||
      0 2px 4px rgba(0, 0, 0, 0.1),
 | 
			
		||||
      0 4px 8px rgba(0, 0, 0, 0.15);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Prevent layout shifts */
 | 
			
		||||
  .flex-grow {
 | 
			
		||||
    flex-grow: 1;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .min-w-0 {
 | 
			
		||||
    min-width: 0;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Ensure container doesn't overflow */
 | 
			
		||||
  .overflow-hidden {
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Touch support for mobile */
 | 
			
		||||
  .touch-active {
 | 
			
		||||
    transform: scale(0.97) !important;
 | 
			
		||||
    opacity: 0.9;
 | 
			
		||||
    transition: transform 0.15s ease-in-out, opacity 0.15s ease-in-out !important;
 | 
			
		||||
    transition:
 | 
			
		||||
      transform 0.15s ease-in-out,
 | 
			
		||||
      opacity 0.15s ease-in-out !important;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Animation for blob */
 | 
			
		||||
  @keyframes blob {
 | 
			
		||||
    0%, 100% {
 | 
			
		||||
    0%,
 | 
			
		||||
    100% {
 | 
			
		||||
      transform: translate(0, 0) scale(1);
 | 
			
		||||
    }
 | 
			
		||||
    25% {
 | 
			
		||||
@@ -530,19 +592,19 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
      transform: translate(-10px, -10px) scale(1.05);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animate-blob {
 | 
			
		||||
    animation: blob 20s infinite ease-in-out;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animation-delay-2000 {
 | 
			
		||||
    animation-delay: 2s;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animation-delay-4000 {
 | 
			
		||||
    animation-delay: 4s;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Animation for underline */
 | 
			
		||||
  @keyframes underline {
 | 
			
		||||
    0% {
 | 
			
		||||
@@ -552,11 +614,11 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
      transform: scaleX(1);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  .animate-underline {
 | 
			
		||||
    animation: underline 1.5s ease-out forwards;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  /* Fix for iOS Safari notch */
 | 
			
		||||
  @supports (padding: max(0px)) {
 | 
			
		||||
    .tag-cloud {
 | 
			
		||||
@@ -571,32 +633,34 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
  // Handle SPA transitions for tags index page
 | 
			
		||||
  function setupSPATransitions() {
 | 
			
		||||
    // Handle all internal links for SPA transitions
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach(link => {
 | 
			
		||||
    document.querySelectorAll('a[href^="/"]').forEach((link) => {
 | 
			
		||||
      // Skip links that are anchor links, external links, or already processed
 | 
			
		||||
      if (link.getAttribute('href').includes('#') || 
 | 
			
		||||
          link.getAttribute('target') === '_blank' || 
 | 
			
		||||
          link.hasAttribute('data-spa-handled')) {
 | 
			
		||||
      if (
 | 
			
		||||
        link.getAttribute('href').includes('#') ||
 | 
			
		||||
        link.getAttribute('target') === '_blank' ||
 | 
			
		||||
        link.hasAttribute('data-spa-handled')
 | 
			
		||||
      ) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // Mark as handled to avoid duplicate listeners
 | 
			
		||||
      link.setAttribute('data-spa-handled', 'true');
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      link.addEventListener('click', (e) => {
 | 
			
		||||
        // Don't handle if modifier keys are pressed (for opening in new tab, etc.)
 | 
			
		||||
        if (e.metaKey || e.ctrlKey || e.shiftKey || e.altKey) {
 | 
			
		||||
          return;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        const targetHref = link.getAttribute('href');
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        // Trigger page transition animation
 | 
			
		||||
        const pageTransition = document.getElementById('page-transition');
 | 
			
		||||
        if (pageTransition) {
 | 
			
		||||
          pageTransition.classList.remove('opacity-0');
 | 
			
		||||
          pageTransition.classList.add('opacity-100');
 | 
			
		||||
          
 | 
			
		||||
 | 
			
		||||
          // Navigate after transition effect
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            window.location.href = targetHref;
 | 
			
		||||
@@ -607,18 +671,18 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Add hover effect for tag cards on touch devices
 | 
			
		||||
    const isTouchDevice = 'ontouchstart' in window || navigator.maxTouchPoints > 0;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    if (isTouchDevice) {
 | 
			
		||||
      const tagCards = document.querySelectorAll('.tag-cloud a');
 | 
			
		||||
      
 | 
			
		||||
      tagCards.forEach(card => {
 | 
			
		||||
 | 
			
		||||
      tagCards.forEach((card) => {
 | 
			
		||||
        card.addEventListener('touchstart', () => {
 | 
			
		||||
          card.classList.add('is-touched');
 | 
			
		||||
        });
 | 
			
		||||
        
 | 
			
		||||
 | 
			
		||||
        card.addEventListener('touchend', () => {
 | 
			
		||||
          setTimeout(() => {
 | 
			
		||||
            card.classList.remove('is-touched');
 | 
			
		||||
@@ -626,23 +690,25 @@ const sortedTags = [...tagObjects].sort((a, b) => b.count - a.count);
 | 
			
		||||
        });
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // Animate tag cards with staggered delay
 | 
			
		||||
    const tagCards = document.querySelectorAll('.tag-cloud a');
 | 
			
		||||
    tagCards.forEach((card, index) => {
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
        card.classList.add('animate-reveal');
 | 
			
		||||
      }, 100 + (index * 50));
 | 
			
		||||
      setTimeout(
 | 
			
		||||
        () => {
 | 
			
		||||
          card.classList.add('animate-reveal');
 | 
			
		||||
        },
 | 
			
		||||
        100 + index * 50
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Initialize on first load
 | 
			
		||||
  document.addEventListener('DOMContentLoaded', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // Re-initialize when content changes via Astro's view transitions
 | 
			
		||||
  document.addEventListener('astro:page-load', setupSPATransitions);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // For compatibility with custom transition system
 | 
			
		||||
  document.addEventListener('page-transition-complete', setupSPATransitions);
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user