Zoeken...  ⌘K GitHub

ImageMosaic image

Mozaïek-layout met één groot beeld en drie kleinere.

/image-mosaic
src/components/image/ImageMosaic.astro
---
interface MosaicImage {
  src?: string;
  alt: string;
  label?: string;
}

interface Props {
  images?: MosaicImage[];
  title?: string;
}

const {
  images = [
    { alt: 'Groot beeld', label: 'Featured project' },
    { alt: 'Klein beeld 1', label: 'Detail 1' },
    { alt: 'Klein beeld 2', label: 'Detail 2' },
    { alt: 'Klein beeld 3', label: 'Detail 3' },
  ],
  title = 'Impressie',
} = Astro.props;
---

<section class="imo">
  {title && <h2 class="imo__title">{title}</h2>}
  <div class="imo__grid">
    {images.slice(0, 4).map((img, i) => (
      <div class:list={['imo__item', i === 0 && 'imo__item--large']}>
        {img.src ? (
          <img class="imo__img" src={img.src} alt={img.alt} />
        ) : (
          <div class="imo__placeholder" aria-label={img.alt}></div>
        )}
        {img.label && <span class="imo__label">{img.label}</span>}
      </div>
    ))}
  </div>
</section>

<style>
  :root {
    --color-accent: #6366f1;
    --color-primary: #0a0a0a;
  }
  .imo { padding: 2rem 0; }
  .imo__title {
    font-size: 2rem;
    font-weight: 700;
    color: var(--color-primary, #0a0a0a);
    margin: 0 0 2rem;
  }
  .imo__grid {
    display: grid;
    grid-template-columns: 2fr 1fr;
    grid-template-rows: auto auto;
    gap: 0.75rem;
  }
  .imo__item {
    position: relative;
    overflow: hidden;
    border-radius: 8px;
    aspect-ratio: 4/3;
  }
  .imo__item--large {
    grid-row: span 2;
    aspect-ratio: auto;
  }
  .imo__img {
    width: 100%;
    height: 100%;
    object-fit: cover;
    transition: transform 0.35s;
  }
  .imo__item:hover .imo__img { transform: scale(1.04); }
  .imo__placeholder {
    width: 100%;
    height: 100%;
    min-height: 150px;
  }
  .imo__item--large .imo__placeholder {
    background: linear-gradient(135deg, #1a2a4a 0%, #3a4a7a 100%);
    min-height: 320px;
  }
  .imo__item:not(.imo__item--large) .imo__placeholder {
    background: linear-gradient(135deg, #e0e0f0 0%, #c8c8e8 100%);
  }
  .imo__item:nth-child(3) .imo__placeholder {
    background: linear-gradient(135deg, #e8f0e0 0%, #c8e0c8 100%);
  }
  .imo__label {
    position: absolute;
    bottom: 0.75rem;
    left: 0.75rem;
    background: rgba(10,10,10,0.65);
    color: #fff;
    font-size: 0.75rem;
    font-weight: 600;
    padding: 0.25rem 0.6rem;
    border-radius: 4px;
  }
  @media (max-width: 640px) {
    .imo__grid { grid-template-columns: 1fr; }
    .imo__item--large { grid-row: auto; }
  }
</style>

Props

Prop Type Default Beschrijving
images { src?: string; alt: string; label?: string }[] Maximaal 4 afbeeldingen; eerste is groot
title string Titel boven het mozaïek

* = verplicht