How to Integrate Three.js & Sanity CMS in my portfolio

How to Integrate Three.js & Sanity CMS in my portfolio

El proceso de integrar Sanity como Content Manager System y Three.js una librerira para renderizar modelos 3D.

📌 Introduction

Have you ever wanted to combine interactive 3D graphics with a modern headless CMS? In this article, I’ll show you step by step how I integrated Three.js and Sanity CMS into my Next.js portfolio, creating an impressive visual experience and a flexible content management system.

🎯 What We’ll Achieve

🛠️ Tech Stack

{
  "frontend": "Next.js 13",
  "3d": "Three.js",
  "cms": "Sanity CMS",
  "styling": "Chakra UI",
  "deployment": "Vercel"
}

🎨 Part 1: Setting Up Three.js

1.1 Installing Dependencies

First, install the necessary dependencies for Three.js:

npm install three @types/three
# O si usas yarn
yarn add three @types/three

1.2 Creating the 3D Component

We’re going to create a component that renders a 3D model of a PC:

// components/voxel-pc.js
import { useState, useEffect, useRef, useCallback } from 'react';
import * as THREE from 'three';
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls';
import { loadGLTFModel } from '../lib/model';
import { PcSpinner, PcContainer } from './voxel-pc-loader';

function easeOutCirc(x) {
  return Math.sqrt(1 - Math.pow(x - 1, 4));
}

const VoxelPc = () => {
  const refContainer = useRef();
  const [loading, setLoading] = useState(true);
  const refRenderer = useRef();
  const urlPcGLB = '/pc.glb';

  const handleWindowResize = useCallback(() => {
    const { current: renderer } = refRenderer;
    const { current: container } = refContainer;
    if (container && renderer) {
      const scW = container.clientWidth;
      const scH = container.clientHeight;
      renderer.setSize(scW, scH);
    }
  }, []);

  useEffect(() => {
    const { current: container } = refContainer;
    if (container) {
      const scW = container.clientWidth;
      const scH = container.clientHeight;

      // Set up renderer
      const renderer = new THREE.WebGLRenderer({
        antialias: true,
        alpha: true,
      });
      renderer.setPixelRatio(window.devicePixelRatio);
      renderer.setSize(scW, scH);
      renderer.outputEncoding = THREE.sRGBEncoding;
      container.appendChild(renderer.domElement);
      refRenderer.current = renderer;

      // Create scene
      const scene = new THREE.Scene();

      // Set up camera
      const target = new THREE.Vector3(0, 0, 0);
      const initialCameraPosition = new THREE.Vector3(
        12 * Math.sin(0.2 * Math.PI),
        6,
        12 * Math.cos(0.2 * Math.PI)
      );

      const scale = scH * 0.003 + 2.2;
      const camera = new THREE.OrthographicCamera(-scale, scale, scale, -scale, 0.01, 50000);
      camera.position.copy(initialCameraPosition);
      camera.lookAt(target);

      // Add lighting
      const ambientLight = new THREE.AmbientLight(0xcccccc, Math.PI);
      scene.add(ambientLight);

      // Set up controls
      const controls = new OrbitControls(camera, renderer.domElement);
      controls.autoRotate = true;
      controls.target = target;

      // Load 3D model
      loadGLTFModel(scene, urlPcGLB, {
        receiveShadow: false,
        castShadow: false,
      })
        .then(() => {
          animate();
          setLoading(false);
        })
        .catch(error => {
          console.error('Error loading pc.glb:', error);
          setLoading(false);
        });

      // Animation loop
      let req = null;
      let frame = 0;
      const animate = () => {
        req = requestAnimationFrame(animate);
        frame = frame <= 100 ? frame + 1 : frame;

        if (frame <= 100) {
          const p = initialCameraPosition;
          const rotSpeed = -easeOutCirc(frame / 120) * Math.PI * 20;

          camera.position.y = 6;
          camera.position.x = p.x * Math.cos(rotSpeed) + p.z * Math.sin(rotSpeed);
          camera.position.z = p.z * Math.cos(rotSpeed) - p.x * Math.sin(rotSpeed);
          camera.lookAt(target);
        } else {
          controls.update();
        }

        renderer.render(scene, camera);
      };

      // Cleanup
      return () => {
        cancelAnimationFrame(req);
        renderer.domElement.remove();
        renderer.dispose();
      };
    }
  }, []);

  // Handle resize
  useEffect(() => {
    window.addEventListener('resize', handleWindowResize, false);
    return () => {
      window.removeEventListener('resize', handleWindowResize, false);
    };
  }, [handleWindowResize]);

  return <PcContainer ref={refContainer}>{loading && <PcSpinner />}</PcContainer>;
};

export default VoxelPc;

1.3 GLTF Model Loader

We create a helper to load 3D models:

// lib/model.js
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader';
import { DRACOLoader } from 'three/addons/loaders/DRACOLoader.js';

const draco = new DRACOLoader();
draco.setDecoderConfig({ type: 'js' });
draco.setDecoderPath('https://www.gstatic.com/draco/v1/decoders/');

export function loadGLTFModel(scene, glbPath, options = { receiveShadow: true, castShadow: true }) {
  const { receiveShadow, castShadow } = options;
  return new Promise((resolve, reject) => {
    const loader = new GLTFLoader();
    loader.setDRACOLoader(draco);

    loader.load(
      glbPath,
      gltf => {
        const obj = gltf.scene;
        obj.name = 'pc';
        obj.position.set(0, 0, 0);
        obj.receiveShadow = receiveShadow;
        obj.castShadow = castShadow;

        // Escalar el modelo
        obj.scale.setScalar(4.5);

        scene.add(obj);

        obj.traverse(function (child) {
          if (child.isMesh) {
            child.castShadow = castShadow;
            child.receiveShadow = receiveShadow;
          }
        });
        resolve(obj);
      },
      undefined,
      error => reject(error)
    );
  });
}

1.4 Loading Component

// components/voxel-pc-loader.js
import { forwardRef } from 'react';
import { Box, Spinner } from '@chakra-ui/react';

export const PcSpinner = () => (
  <Box position="absolute" top="50%" left="50%" transform="translate(-50%, -50%)">
    <Spinner size="xl" color="orange.400" />
  </Box>
);

export const PcContainer = forwardRef(({ children }, ref) => (
  <Box
    ref={ref}
    className="voxel-pc"
    m="auto"
    mt={['20px', '-60px', '-120px']}
    mb={['-40px', '-140px', '-200px']}
    w={[280, 480, 640]}
    h={[280, 480, 640]}
    position="relative"
  >
    {children}
  </Box>
));

PcContainer.displayName = 'PcContainer';

🏗️ Part 2: Setting Up Sanity CMS

2.1 Installing Sanity

npm install @sanity/client sanity @sanity/image-url

2.2 Client Configuration

// lib/sanity.js
import { createClient } from '@sanity/client';
import imageUrlBuilder from '@sanity/image-url';

const config = {
  projectId: 'tu-project-id', // Obtén esto desde sanity.io
  dataset: 'production',
  apiVersion: '2024-01-01',
  useCdn: process.env.NODE_ENV === 'production',
  token: process.env.SANITY_API_TOKEN,
};

const client = createClient(config);
const builder = imageUrlBuilder(client);

export const urlFor = source => {
  if (!source) return null;
  return builder.image(source).auto('format').fit('max');
};

export const getOptimizedImageUrl = (source, width = 800, height = null) => {
  if (!source) return null;
  let imageBuilder = builder.image(source).auto('format').fit('max').width(width);
  if (height) {
    imageBuilder = imageBuilder.height(height);
  }
  return imageBuilder.url();
};

export default client;

// Optimized queries
export const queries = {
  projects: `*[_type == "project"] | order(order asc) {
    _id,
    title,
    slug,
    description,
    year,
    platform,
    stack[0...3],
    users,
    roi,
    growth,
    order,
    featured,
    type,
    image {
      asset->{
        _id,
        url,
        metadata {
          dimensions {
            width,
            height
          },
          lqip
        }
      },
      alt
    }
  }`,

  project: `*[_type == "project" && slug.current == $slug][0] {
    _id,
    title,
    slug,
    description,
    year,
    platform,
    stack,
    features,
    users,
    roi,
    growth,
    website,
    image {
      asset->{
        _id,
        url,
        metadata {
          dimensions {
            width,
            height
          },
          lqip
        }
      },
      alt
    },
    gallery[] {
      asset->{
        _id,
        url,
        metadata {
          dimensions {
            width,
            height
          },
          lqip
        }
      },
      alt,
      caption
    },
    content,
    type,
    order
  }`,
};

2.3 Environment Variables

# .env.local
NEXT_PUBLIC_SANITY_PROJECT_ID=tu-project-id
NEXT_PUBLIC_SANITY_DATASET=production
SANITY_API_TOKEN=tu-api-token

2.4 Project Schema

// sanity/schemas/project.js
export default {
  name: 'project',
  title: 'Projects',
  type: 'document',
  fields: [
    {
      name: 'title',
      title: 'Title',
      type: 'string',
      validation: Rule => Rule.required(),
    },
    {
      name: 'slug',
      title: 'Slug',
      type: 'slug',
      options: {
        source: 'title',
        maxLength: 96,
      },
      validation: Rule => Rule.required(),
    },
    {
      name: 'description',
      title: 'Description',
      type: 'text',
      validation: Rule => Rule.required(),
    },
    {
      name: 'image',
      title: 'Thumbnail Image',
      type: 'image',
      options: {
        hotspot: true,
      },
      fields: [
        {
          name: 'alt',
          type: 'string',
          title: 'Alternative text',
        },
      ],
    },
    {
      name: 'gallery',
      title: 'Image Gallery',
      type: 'array',
      of: [
        {
          type: 'image',
          options: { hotspot: true },
          fields: [
            {
              name: 'alt',
              type: 'string',
              title: 'Alternative text',
            },
            {
              name: 'caption',
              type: 'string',
              title: 'Caption',
            },
          ],
        },
      ],
    },
    {
      name: 'stack',
      title: 'Tech Stack',
      type: 'array',
      of: [{ type: 'string' }],
      options: {
        layout: 'tags',
      },
    },
    {
      name: 'year',
      title: 'Year',
      type: 'number',
      validation: Rule => Rule.required(),
    },
    {
      name: 'type',
      title: 'Project Type',
      type: 'string',
      options: {
        list: [
          { title: 'Main Project', value: 'main' },
          { title: 'Collaboration', value: 'collaboration' },
          { title: 'Side Project', value: 'side' },
        ],
      },
      initialValue: 'main',
    },
    {
      name: 'featured',
      title: 'Featured',
      type: 'boolean',
      initialValue: false,
    },
    {
      name: 'users',
      title: 'Users/Downloads',
      type: 'string',
      description: 'e.g., "10K+ users", "5K downloads"',
    },
    {
      name: 'roi',
      title: 'ROI/Impact',
      type: 'string',
      description: 'e.g., "30% increase in sales"',
    },
    {
      name: 'website',
      title: 'Website URL',
      type: 'url',
    },
    {
      name: 'order',
      title: 'Display Order',
      type: 'number',
      validation: Rule => Rule.required(),
    },
  ],
  preview: {
    select: {
      title: 'title',
      media: 'image',
      subtitle: 'year',
    },
  },
  orderings: [
    {
      title: 'Display Order',
      name: 'orderAsc',
      by: [{ field: 'order', direction: 'asc' }],
    },
  ],
};

🖼️ Part 3: Image Optimization

3.1 Optimized Image Component

// components/sanity-image.js
import Image from 'next/image';
import { urlFor } from '../lib/sanity';

const SanityImage = ({
  image,
  alt = '',
  width = 800,
  height = 600,
  className = '',
  priority = false,
  quality = 85,
  ...props
}) => {
  if (!image?.asset) {
    return (
      <div
        className={`bg-gray-100 flex items-center justify-center ${className}`}
        style={{ width, height }}
      >
        <span className="text-gray-400 text-sm">No image</span>
      </div>
    );
  }

  const imageUrl = urlFor(image).width(width).height(height).quality(quality).auto('format').url();

  const blurDataURL =
    image.asset.metadata?.lqip ||
    '';

  return (
    <Image
      src={imageUrl}
      alt={alt || image.alt || 'Image'}
      width={width}
      height={height}
      className={className}
      priority={priority}
      placeholder="blur"
      blurDataURL={blurDataURL}
      {...props}
    />
  );
};

export default SanityImage;

3.2 Next.js Configuration

// next.config.js
module.exports = {
  reactStrictMode: true,
  swcMinify: true,
  images: {
    domains: ['cdn.sanity.io'],
  },
};

🔗 Part 4: Integrating Everything

4.1 Projects Page

// pages/works.js
import { Container, Heading, SimpleGrid, Divider } from '@chakra-ui/react';
import Layout from '../components/layouts/article';
import Section from '../components/section';
import { WorkGridItem } from '../components/grid-item';
import client, { queries, urlFor } from '../lib/sanity';

const Works = ({ projects = [] }) => {
  if (projects.length === 0) {
    return <div>No projects found</div>;
  }

  const mainProjects = projects.filter(p => p.type === 'main');
  const collaborations = projects.filter(p => p.type === 'collaboration');
  const sideProjects = projects.filter(p => p.type === 'side');

  return (
    <Layout title="Works">
      <Container>
        <Heading as="h3" fontSize={20} mb={4}>
          Works
        </Heading>

        <SimpleGrid columns={[1, 1, 2]} gap={6}>
          {mainProjects.map((project, index) => {
            const thumbnail = project.image?.asset?.url
              ? urlFor(project.image).width(400).height(225).auto('format').quality(85).url()
              : '/fallback-image.jpg';

            return (
              <Section key={project._id} delay={index * 0.1}>
                <WorkGridItem id={project.slug.current} title={project.title} thumbnail={thumbnail}>
                  {project.description}
                </WorkGridItem>
              </Section>
            );
          })}
        </SimpleGrid>

        {collaborations.length > 0 && (
          <>
            <Section delay={0.2}>
              <Divider my={6} />
              <Heading as="h3" fontSize={20} mb={4}>
                Collaborations
              </Heading>
            </Section>
            <SimpleGrid columns={[1, 1, 2]} gap={6}>
              {collaborations.map((project, index) => (
                <Section key={project._id} delay={0.3 + index * 0.1}>
                  <WorkGridItem
                    id={project.slug.current}
                    title={project.title}
                    thumbnail={urlFor(project.image).width(400).height(225).url()}
                  >
                    {project.description}
                  </WorkGridItem>
                </Section>
              ))}
            </SimpleGrid>
          </>
        )}
      </Container>
    </Layout>
  );
};

export async function getServerSideProps() {
  try {
    const projects = await client.fetch(queries.projects);
    return {
      props: { projects },
    };
  } catch (error) {
    console.error('Failed to fetch projects from Sanity:', error);
    return {
      props: { projects: [] },
    };
  }
}

export default Works;

4.2 Home Page with 3D Model

// pages/index.js
import NextLink from 'next/link';
import {
  Link,
  Container,
  Heading,
  Box,
  Button,
  List,
  ListItem,
  useColorModeValue,
  chakra,
} from '@chakra-ui/react';
import { ChevronRightIcon, EmailIcon } from '@chakra-ui/icons';
import Paragraph from '../components/paragraph';
import { BioSection, BioYear } from '../components/bio';
import Layout from '../components/layouts/article';
import Section from '../components/section';
import { GitHubIcon, TwitterIcon, LinkedInIcon } from '../components/icons';
import Image from 'next/image';
import { IoLogoTwitter, IoLogoInstagram, IoLogoGithub, IoLogoLinkedin } from 'react-icons/io5';
import VoxelPc from '../components/voxel-pc';

const ProfileImage = chakra(Image, {
  shouldForwardProp: prop => ['width', 'height', 'src', 'alt'].includes(prop),
});

const Home = () => (
  <Layout>
    <Container>
      <Box
        borderRadius="lg"
        mb={6}
        p={3}
        textAlign="center"
        bg={useColorModeValue('whiteAlpha.500', 'whiteAlpha.200')}
        css={{ backdropFilter: 'blur(10px)' }}
      >
        Hello, I&apos;m an indie developer based in Rosario!
      </Box>

      <Box display={{ md: 'flex' }}>
        <Box flexGrow={1}>
          <Heading as="h2" variant="page-title">
            Tomás Maritano
          </Heading>
          <p>Digital Craftsman ( Developer / Designer / Entrepreneur )</p>
        </Box>
        <Box flexShrink={0} mt={{ base: 4, md: 0 }} ml={{ md: 6 }} textAlign="center">
          <Box
            borderColor="whiteAlpha.800"
            borderWidth={2}
            borderStyle="solid"
            w="100px"
            h="100px"
            display="inline-block"
            borderRadius="full"
            overflow="hidden"
          >
            <ProfileImage
              src="/images/tomy.jpg"
              alt="Profile image"
              borderRadius="full"
              width="100"
              height="100"
            />
          </Box>
        </Box>
      </Box>

      <Section delay={0.1}>
        <Heading as="h3" variant="section-title">
          Work
        </Heading>
        <Paragraph>
          Tomás is an indie developer with 7+ years scaling from Frontend Developer to Product
          Manager at companies like Wolt, Unicoin, and Valere. He specializes in end-to-end product
          development and team leadership.
        </Paragraph>
        <Box align="center" my={4}>
          <Button
            as={NextLink}
            href="/works"
            scroll={false}
            rightIcon={<ChevronRightIcon />}
            colorScheme="teal"
          >
            My portfolio
          </Button>
        </Box>
      </Section>

      <Section delay={0.2}>
        <Heading as="h3" variant="section-title">
          Bio
        </Heading>
        <BioSection>
          <BioYear>1995</BioYear>
          Born in Rosario, Argentina.
        </BioSection>
        <BioSection>
          <BioYear>2018</BioYear>
          Started as Frontend Developer
        </BioSection>
        <BioSection>
          <BioYear>2022</BioYear>
          Promoted to Product Manager at Wolt
        </BioSection>
        <BioSection>
          <BioYear>2025 to present</BioYear>
          Working as an indie developer
        </BioSection>
      </Section>

      <Section delay={0.3}>
        <Heading as="h3" variant="section-title">
          On the web
        </Heading>
        <List>
          <ListItem>
            <Link href="https://github.com/tomymaritano" target="_blank">
              <Button variant="ghost" colorScheme="teal" leftIcon={<IoLogoGithub />}>
                @tomymaritano
              </Button>
            </Link>
          </ListItem>
          <ListItem>
            <Link href="https://twitter.com/hacklabdog" target="_blank">
              <Button variant="ghost" colorScheme="teal" leftIcon={<IoLogoTwitter />}>
                @hacklabdog
              </Button>
            </Link>
          </ListItem>
          <ListItem>
            <Link href="https://linkedin.com/in/tomymaritano" target="_blank">
              <Button variant="ghost" colorScheme="teal" leftIcon={<IoLogoLinkedin />}>
                @tomymaritano
              </Button>
            </Link>
          </ListItem>
        </List>
      </Section>

      <Section delay={0.3}>
        <Box align="center" my={4}>
          <VoxelPc />
        </Box>
      </Section>
    </Container>
  </Layout>
);

export default Home;
export { getServerSideProps } from '../components/chakra';

⚡ Part 5: Optimizations and Performance

5.1 Lazy Loading 3D Components

// components/lazy-voxel.js
import dynamic from 'next/dynamic';
import { PcSpinner, PcContainer } from './voxel-pc-loader';

const LazyVoxelPc = dynamic(() => import('./voxel-pc'), {
  ssr: false,
  loading: () => (
    <PcContainer>
      <PcSpinner />
    </PcContainer>
  ),
});

export default LazyVoxelPc;

5.2 GROQ Query Optimization

// Queries optimizadas para diferentes casos de uso
export const optimizedQueries = {
  // Para listados (solo datos esenciales)
  projectsListing: `*[_type == "project"] | order(order asc) {
    _id,
    title,
    slug,
    description,
    type,
    "thumbnail": image.asset->url
  }[0...12]`, // Limitar resultados

  // Para página individual (datos completos)
  projectDetail: `*[_type == "project" && slug.current == $slug][0] {
    ...,
    "imageUrl": image.asset->url,
    "galleryImages": gallery[].asset->url
  }`,

  // Para sitemap (solo slugs)
  projectSlugs: `*[_type == "project"].slug.current`,
};

5.3 Implementing ISR (Incremental Static Regeneration)

// pages/works/[slug].js
export async function getStaticProps({ params }) {
  const project = await client.fetch(queries.project, { slug: params.slug });

  if (!project) {
    return { notFound: true };
  }

  return {
    props: { project },
    revalidate: 60, // Revalidar cada minuto
  };
}

export async function getStaticPaths() {
  const slugs = await client.fetch(`*[_type == "project"].slug.current`);

  return {
    paths: slugs.map(slug => ({ params: { slug } })),
    fallback: 'blocking',
  };
}

🚀 Part 6: Deployment and Configuration

6.1 Vercel Configuration

// vercel.json
{
  "functions": {
    "pages/api/revalidate.js": {
      "maxDuration": 10
    }
  },
  "env": {
    "NEXT_PUBLIC_SANITY_PROJECT_ID": "@sanity-project-id",
    "SANITY_API_TOKEN": "@sanity-api-token"
  }
}

6.2 Auto Revalidation Webhook

// pages/api/revalidate.js
export default async function handler(req, res) {
  if (req.query.secret !== process.env.REVALIDATE_SECRET) {
    return res.status(401).json({ message: 'Invalid token' });
  }

  try {
    await res.revalidate('/works');
    return res.json({ revalidated: true });
  } catch (err) {
    return res.status(500).send('Error revalidating');
  }
}

6.3 Sanity Studio Configuration

// sanity.config.js
import { defineConfig } from 'sanity';
import { deskTool } from 'sanity/desk';
import { visionTool } from '@sanity/vision';
import { schemaTypes } from './schemas';

export default defineConfig({
  name: 'default',
  title: 'Portfolio CMS',
  projectId: process.env.NEXT_PUBLIC_SANITY_PROJECT_ID,
  dataset: 'production',

  plugins: [deskTool(), visionTool()],

  schema: {
    types: schemaTypes,
  },

  tools: prev => {
    if (process.env.NODE_ENV === 'development') {
      return prev;
    }
    return prev.filter(tool => tool.name !== 'vision');
  },
});

📊 Part 7: Metrics and Monitoring

7.1 Performance monitoring

// lib/analytics.js
export const trackModelLoad = (loadTime, modelSize) => {
  if (typeof window !== 'undefined' && window.gtag) {
    window.gtag('event', '3d_model_loaded', {
      event_category: 'Performance',
      event_label: 'PC Model',
      value: Math.round(loadTime),
      custom_parameter_1: modelSize,
    });
  }
};

export const trackCMSQuery = (queryType, responseTime) => {
  if (typeof window !== 'undefined' && window.gtag) {
    window.gtag('event', 'cms_query', {
      event_category: 'CMS',
      event_label: queryType,
      value: Math.round(responseTime),
    });
  }
};

7.2 Error Boundary for 3D Components

// components/error-boundary.js
import React from 'react';
import { Box, Text, Button } from '@chakra-ui/react';

class ModelErrorBoundary extends React.Component {
  constructor(props) {
    super(props);
    this.state = { hasError: false };
  }

  static getDerivedStateFromError(error) {
    return { hasError: true };
  }

  componentDidCatch(error, errorInfo) {
    console.error('3D Model Error:', error, errorInfo);
  }

  render() {
    if (this.state.hasError) {
      return (
        <Box textAlign="center" p={8}>
          <Text mb={4}>Could not load 3D model</Text>
          <Button onClick={() => this.setState({ hasError: false })}>Try Again</Button>
        </Box>
      );
    }

    return this.props.children;
  }
}

export default ModelErrorBoundary;

🎯 Conclusions and Best Practices

✅ What We Achieved

  1. Impressive visual experience with interactive 3D models
  2. Completely free CMS with Sanity
  3. Optimized performance with lazy loading and ISR
  4. Scalability to grow with your portfolio
  5. SEO friendly with server-side rendering

🔧 Best Practices Implemented

📈 Next Steps

  1. Implement PWA for improved mobile experience
  2. Add animations with Framer Motion
  3. A/B testing for 3D components
  4. Internationalization (i18n)
  5. Advanced analytics with event tracking

🔗 Useful Resources


💻 Full Code

All the code from this article is available on my GitHub.

Did you enjoy this tutorial? Follow me on Twitter for more content about web development and technology.


This article was created as part of my experience developing my personal portfolio. If you have any questions or suggestions, feel free to reach out.

Tags: three.js sanity-cms nextjs react 3d-web portfolio javascript webgl


title: ‘How I Built My Link-in-Bio with Vite, React & Framer Motion’ date: ‘2025-06-02’ excerpt: ‘Here’s the process of creating my animated link-in-bio using Vite, React, Tailwind CSS, and Framer Motion.’ coverImage: ‘/images/vite.png’


I wanted a minimal, animated link-in-bio that I could easily host and update myself — without relying on third-party services.

Here’s my process, tools, and what I learned along the way.


My workspace on the go

How I Built It — Step by Step

Here’s the exact process I followed to build this link-in-bio page:

1. Installed the Environment

First, I set up a new Vite + React project:

npm create vite@latest link-in-bio -- --template react
cd link-in-bio

Then I added the main dependencies:

npm install tailwindcss postcss autoprefixer framer-motion
npx tailwindcss init -p

And configured Tailwind in tailwind.config.js and index.css.

2. Created the Core Components

Next, I broke the UI into small reusable pieces:

Each component was styled using Tailwind utility classes.

3. Integrated Animations with Framer Motion

I used Framer Motion for smooth interactions:

4. Added a Dark Mode Toggle

I created a simple ToggleDarkMode.jsx:

Finally

I wrapped the main layout in a motion.div so that:

This subtle polish made the app feel more dynamic and enjoyable to use.


Why I Built It Myself

Most link-in-bio tools look the same and charge a subscription fee.
I built my own because:


Tech Stack

Here’s what I used in this project:


What I Learned