How to Build a Simple Portfolio Website With HTML and CSS

While browsing through some of my old projects, I stumbled upon a portfolio I worked on a while back. The funny thing? I never used it. So, I decided to give it a fresh look and thought, why not share it with you? As a developer, having a portfolio i...

Jan 14, 2025 - 13:36
How to Build a Simple Portfolio Website With HTML and CSS

While browsing through some of my old projects, I stumbled upon a portfolio I worked on a while back. The funny thing? I never used it. So, I decided to give it a fresh look and thought, why not share it with you?

As a developer, having a portfolio is important. It’s your personal space to showcase your skills, share projects, and impress potential employers. Whether you’re aiming to build an online presence, share your work with friends, or take your career to the next level, a portfolio is the perfect way to highlight what you can do.

In this article, I’ll guide you through building a simple portfolio website using just HTML and CSS. By the end, you’ll have a fully functional portfolio site ready to share with anyone.

What the Finished Portfolio Looks Like:

Before we get started, here’s a preview of what your portfolio will look like once we’re done:

d8c88f72-5e79-466d-9fe0-d8814be7f1cb

If you’d prefer to skip ahead, you can get the full code on GitHub.

Here’s what we’ll cover:

Project Overview and Setup

Before we start coding, let’s take a look at the basic structure of the project. Here's how everything will be organized:

/assets
  - background_image.jpg
  - user.png
  - icon-github.svg
  - icon-twitter.svg
  - logo1.png
  - logo2.png
  - logo3.png
index.html
styles.css

This folder structure keeps things neat and organized. The /assets folder will hold all of your images and icons that will be used throughout the website, while the index.html file will contain the structure of your portfolio, and styles.css will control how everything looks.

Setting Up the HTML File

Let's start with the basic HTML structure. Open your index.html file and add the following code:

html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <link rel="stylesheet" href="styles.css" />
    <title>Spruce - Dev Portfoliotitle>
  head>
  <body>
    
  body>
html>

Here’s a breakdown of what this code does:

  1. DOCTYPE: This tells the browser that this is an HTML5 document.

  2. meta charset: Defines the character encoding, which helps display text properly.

  3. meta viewport: Ensures the website is responsive on different screen sizes.

  4. link to styles.css: Links the styles.css file for styling your website.

  5. title: This is the text that appears in the browser tab when you open the website.

Now that our HTML file is set up, we’ll move on to the CSS.

Setting Up the CSS File

Let’s now add some basic styling using a CSS reset. This helps eliminate browser inconsistencies by normalizing styles across different browsers.

Here’s the basic CSS reset you’ll need:

/**
 * CORE CONFIG
 * This powers everything from utility class generation to breakpoints
 * to enabling/disabling pre-built components/utilities.
 */
/* Box sizing rules */
*,
*::before,
*::after {
  box-sizing: border-box;
  font-family: sans-serif;
}

/* Remove default margin */
body,
h1,
h2,
h3,
h4,
p,
figure,
blockquote,
dl,
dd {
  margin: 0;
}

/* Remove list styles on ul, ol elements with a list role, which suggests default styling will be removed */
ul[role="list"],
ol[role="list"] {
  list-style: none;
  padding: 0;
}

/* Set core root defaults */
html:focus-within {
  scroll-behavior: smooth;
}

/* Set core body defaults */
body {
  min-height: 100vh;
  text-rendering: optimizeSpeed;
  letter-spacing: -0.01em;
}

/* All elements that don't have a class get default styles */
a:not([class]) {
  text-decoration-skip-ink: auto;
}

/* Make images easier to work with */
img,
picture {
  max-width: 100%;
  display: block;
}

/* Inherit fonts for inputs and buttons */
input,
button,
textarea,
select {
  font: inherit;
}

@media (prefers-reduced-motion: reduce) {
  html:focus-within {
    scroll-behavior: auto;
  }
}

:focus {
  outline: 2px dashed #00ff19;
  outline-offset: 0.25rem;
}

main:focus {
  outline: none;
}

This reset clears any default margins and paddings, ensuring that your layout remains consistent across browsers. You can find more info about what the code does in the comments.

Sizing and Spacing with Utopia

For spacing and typography, we’ll use a flexible system from Utopia, a tool designed to make responsive typography easier to implement. Here’s the CSS that will give us scalable, adjustable spacing and font sizes:

/* @link https://utopia.fyi/type/calculator?c=320,18,1.2,1240,20,1.25,5,2,&s=0.75|0.5|0.25,1.5|2|3|4|6,s-l&g=s,l,xl,12 */

:root {
  --step--2: clamp(0.7813rem, 0.7747rem + 0.0326vw, 0.8rem);
  --step--1: clamp(0.9375rem, 0.9158rem + 0.1087vw, 1rem);
  --step-0: clamp(1.125rem, 1.0815rem + 0.2174vw, 1.25rem);
  --step-1: clamp(1.35rem, 1.2761rem + 0.3696vw, 1.5625rem);
  --step-2: clamp(1.62rem, 1.5041rem + 0.5793vw, 1.9531rem);
  --step-3: clamp(1.944rem, 1.771rem + 0.8651vw, 2.4414rem);
  --step-4: clamp(2.3328rem, 2.0827rem + 1.2504vw, 3.0518rem);
  --step-5: clamp(2.7994rem, 2.4462rem + 1.7658vw, 3.8147rem);
}

/* @link https://utopia.fyi/space/calculator?c=320,18,1.2,1240,20,1.25,5,2,&s=0.75|0.5|0.25,1.5|2|3|4|6,s-l&g=s,l,xl,12 */

:root {
  --space-3xs: clamp(0.3125rem, 0.3125rem + 0vw, 0.3125rem);
  --space-2xs: clamp(0.5625rem, 0.5408rem + 0.1087vw, 0.625rem);
  --space-xs: clamp(0.875rem, 0.8533rem + 0.1087vw, 0.9375rem);
  --space-s: clamp(1.125rem, 1.0815rem + 0.2174vw, 1.25rem);
  --space-m: clamp(1.6875rem, 1.6223rem + 0.3261vw, 1.875rem);
  --space-l: clamp(2.25rem, 2.163rem + 0.4348vw, 2.5rem);
  --space-xl: clamp(3.375rem, 3.2446rem + 0.6522vw, 3.75rem);
  --space-2xl: clamp(4.5rem, 4.3261rem + 0.8696vw, 5rem);
  --space-3xl: clamp(6.75rem, 6.4891rem + 1.3043vw, 7.5rem);
}

These CSS variables give you flexible, responsive font sizes and spacing, based on the viewport size. For instance, --step-5 corresponds to the largest font size, which you will use for main headings (h1), and smaller sizes like --step-1 for text that's less prominent.

How Typography and Spacing Work Together

To apply these scalable values, we’ll use the following for headings:

h1,
.h1 {
  font-size: var(--step-5);
}

h2,
.h2 {
  font-size: var(--step-4);
}

h3,
.h3 {
  font-size: var(--step-3);
}

h4,
.h4 {
  font-size: var(--step-2);
}

h5,
.h5 {
  font-size: var(--step-1);
}

Here, the h1 will be the largest size (using --step-5), and h2 will be a little smaller (--step-4), continuing down to h5. These variables ensure the text adjusts to different screen sizes and maintains consistent visual hierarchy.

Defining Constants for Layouts and Spacing

We can also define some utility classes and wrappers for more consistent layouts across the website. Here's how we can define some constants for layout styling:

/* define some constants */
.flex {
  display: flex;
  align-items: center;
  gap: var(--space-s);
}
@media (max-width: 468px) {
  .flex {
    flex-wrap: wrap;
  }
}
.flex_center {
  justify-content: center;
}
.flex_around {
  justify-content: space-around;
}
.flex_between {
  justify-content: space-between;
}
@media (max-width: 468px) {
  .flex_around {
    justify-content: center;
  }
  .flex_between {
    justify-content: center;
  }
}
.text_center {
  text-align: center;
}
.text_right {
  text-align: right;
}

/* define spacing */
.padding-top-3xs {
  padding: var(--space-3xs);
}

These classes allow you to quickly apply flexible layouts and spacing throughout your design. The .flex class will apply Flexbox, while .flex_center centers the content. There are also responsive adjustments for smaller screens like wrapping flex items when necessary.

Creating the Wrappers

To help structure your layout neatly and consistently, we’ll use wrappers. Wrappers ensure that content is centered and doesn’t stretch too far on wide screens:

/* create the wrappers */
.wrapper {
  width: 100%;
  max-width: 1240px;
  margin: 0 auto;
}
.wrapper_inner {
  width: 100%;
  max-width: 800px;
  margin: 0 auto;
}

The .wrapper class ensures that the content doesn’t get too wide, while .wrapper_inner adds another layer of control for inner sections like a main content area or smaller containers.

How to Build the Header and Hero Section

Now that our project structure is ready, let’s get into building the first major sections: the Header and Hero. These sections are your site’s first impression, giving visitors a sense of who you are while offering easy navigation.

HTML for the Header and Hero

Here’s the HTML structure for both sections:

html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <link rel="stylesheet" href="styles.css" />
    <title>Spruce - Dev Portfoliotitle>
  head>
  <body>
    <header class="site_header">
      <div class="wrapper_inner">
        <nav class="site_header_nav">
          <ul role="list" class="flex flex_center">
            <li><a href="#0">HOMEa>li>
            <li><a href="#about">ABOUTa>li>
            <li><a href="#work">WORKa>li>
            <li><a href="#contact">CONTACTa>li>
          ul>
        nav>
      div>
    header>
    <main>
      <section id="hero">
        <div class="wrapper">
          <div class="wrapper_inner">
            <div class="text_center hero_content">
              <h1 class="h2">Spruce Emmanuelh1>
              <p class="h5">WEB DEVELOPER && TECHNICAL WRITERp>
            div>
          div>
          <ul role="list" class="flex flex_between hero_social">
            <address>Port Harcourt, Nigeriaaddress>
            <ul role="list" class="flex">
              <li>Find me onlineli>
              <li>
                <a href="https://github.com/iamspruce" target="_blank">
                  <img src="/assets/icon-github.svg" alt="GitHub" width="24" height="24" />
                a>
              li>
              <li>
                <a href="https://x.com/sprucekhalifa" target="_blank">
                  <img src="/assets/icon-twitter.svg" alt="Twitter" width="24" height="24" />
                a>
              li>
            ul>
          ul>
        div>
      section>
    main>
  body>
html>

Header breakdown:

  • Navigation: The navbar lets users quickly jump to key sections like Home, About, Work, and Contact. It’s centered using the .flex and .flex_center classes, making it clean and accessible.

  • Flex layout: Links are laid out horizontally using .flex for a smooth, modern look.

Hero breakdown:

  • Hero content: This is where you introduce yourself. Your name and title are centered with padding to give it space and clarity.

  • Hero social: Below your bio, we’ve added your location and social links. These are aligned with .flex to keep everything neat and easy to navigate.

CSS for Header and Hero

Here’s the CSS to style these sections:

Header styling:

/* Style the header */
.site_header {
  padding: var(--space-s) 0;
  background: linear-gradient(rgba(26, 26, 64, 0.9), rgba(26, 26, 64, 0.9));
}

.site_header_nav ul li a {
  text-decoration: none;
  color: #fff;
}
  • .site_header: Adds padding and a dark gradient for a sleek background.

  • .site_header_nav ul li a: Removes link underlines and makes the text white for a modern, bold look.

Hero styling:

/* Hero section styling */
.hero_content {
  padding: var(--space-3xl) 0;
}

.hero_content h1 {
  margin-bottom: var(--space-s);
}

.hero_social {
  padding-top: var(--space-3xl);
}

/* General styling for the hero section */
section {
  padding: var(--space-2xl) var(--space-l);
}

section#hero {
  background: linear-gradient(rgba(26, 26, 64, 0.9), rgba(26, 26, 64, 0.9)),
    url("/assets/background_image.jpg");
  padding-bottom: 4px;
  background-size: cover;
  background-position: top;
  background-attachment: scroll;
  color: #fafafa;
  --color-accent: #8a2be2;
}
  • .hero_content: Adds padding to keep the content spacious.

  • .hero_social: Makes room for the social media links to stand out.

  • section#hero: Sets a stylish gradient background with an image, ensuring the hero section feels engaging and visually striking.

Here’s what it looks like at this point:

Portfolio after adding the hero and header sections

How to Build the About Section

The About Section lets you introduce yourself and share your professional journey, giving visitors a glimpse of who you are and what you do.

HTML for the About Section

Here’s the HTML structure for the About Section:

<section id="about">
  <div class="wrapper">
    <div class="text_center">
      <h2>
        I am a web developer and technical writer. I write articles on web development and programming.
      h2>
    div>
    <div class="wrapper_inner">
      <div class="about_content flex flex_around">
        <figure class="about_figure text_center">
          <img width="200px" height="200px" src="/assets/user.png" alt="Developer" class="about_img" />
          <figcaption class="about_caption">Spruce ❤️figcaption>
        figure>
        <p class="h5">
          Born in Nigeria, I've spent 4+ years building with and writing about JavaScript. I help companies market their products and services through technical writing.
        p>
      div>
    div>
  div>
section>

Breakdown of the About section:

  • Introduction: A short and clear statement about your role as a developer.

  • About content: The profile image and bio are positioned side by side with a flex container for balance.

  • Figure and image: The profile image is displayed within a

    tag, with a caption below it.
  • Text: A brief bio explaining your background and expertise.

CSS for the About Section

Here’s the CSS styling for the About section:

About section styling:

/* About section */
.about_content {
  padding: var(--space-xl) 0;
  transform: skewX(-10deg);
}

@media (max-width: 468px) {
  .about_content {
    transform: skewX(0deg);
  }
}

.about_figure {
  padding: var(--space-xl) 0;
  box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
  border: 2px solid #ccc;
  display: inline-block;
  padding: 5px;
}

.about_img {
  width: 300px;
  height: 200px;
  object-fit: cover;
}

.about_caption {
  padding: var(--space-3xs) 0;
}

section#about {
  background-color: #ffffff;
  color: #333;
  --color-accent: #ffd700;
}

Breakdown of the CSS:

  • Skew effect: The .about_content class gives the section a dynamic look with a slight skew, which is removed on small screens.

  • Figure styling: The profile image gets padding and a subtle box shadow to help it stand out.

  • Image styling: The image fills the space proportionally, ensuring it doesn't distort.

  • Caption styling: Padding around the caption provides better spacing.

About section preview:

This is what your About Section will look like after applying these styles:

Portfolio after adding the About section

How to Build the Work Section

The Work Section is where you can showcase the companies or projects you’ve worked with or contributed to. This section helps to demonstrate your experience and the impact you've made, while providing visitors with visual examples of your professional background.

HTML for the Work Section

Here’s the HTML structure for the Work Section:

<section id="work">
  <div class="wrapper_inner">
    <div class="text_center">
      <h4 class="h5">
        So far I've helped 20+ companies build their web presence and
        improve their technical documentation.
      h4>
      <ul role="list" class="work_logos flex flex_center">
        <li>
          <img src="/assets/logo1.png" alt="Acme Corp" />
        li>
        <li>
          <img src="/assets/logo2.png" alt="Acme Innovations" />
        li>
        <li>
          <img
            src="/assets/logo1.png"
            alt="Acme Solutions"
            width="50"
            height="50"
          />
        li>
        <li>
          <img src="/assets/logo3.png" alt="Acme Technologies" />
        li>
        <li>
          <img src="/assets/logo1.png" alt="Acme Enterprises" />
        li>
      ul>
    div>
  div>
section>

Breakdown of the Work section:

  • Introduction: The

    element introduces the work experience with a statement highlighting how you've helped companies with their web presence and technical documentation.

  • Logos: The logos of the companies you've worked with are placed inside an unordered list (

      ). Each logo is wrapped in a list item (
    • ) for better structure and layout.
    • Flexbox layout: The logos are positioned in a flexible layout using the flex and flex_center classes. This ensures that the logos are evenly spaced and centered horizontally within the section.

    CSS Styling for the Work Section

    Now let’s add some style to the Work section to make it visually appealing and responsive.

    Work section styling:

    /* work section */
    .work_logos {
      margin-top: var(--space-xl);
    }
    
    section#work {
      background-color: #87ceeb;
      color: #0c0c0c;
      --color-accent: #2e8b57;
    }
    

    How to Build the Contact Section and Footer

    The Contact Section and Footer provide a space for visitors to reach out to you and a final touch for the webpage. It’s essential to have an easy-to-use form for inquiries, along with a footer that includes your copyright information.

    HTML for the Contact Section and Footer

    Here’s the HTML structure for the Contact Section and Footer:

    <section id="contact">
      <div class="wrapper">
        <div class="text_left">
          <p>
            Want to work together? <br />
            I'd love to hear from you.
          p>
        div>
    
        <div class="wrapper_inner">
          <h5 class="h3">Contact Meh5>
          <form action="" method="post" class="contact_form">
            <div class="flex">
              <div class="form_group">
                <label for="name">NAMElabel>
                <input type="text" id="name" name="name" />
              div>
              <div class="form_group">
                <label for="email">EMAILlabel>
                <input type="email" id="email" name="email" />
              div>
            div>
            <div class="form_group">
              <label for="message">Messagelabel>
              <textarea id="message" name="message">textarea>
            div>
            <button type="submit">SEND MESSAGEbutton>
          form>
        div>
      div>
    section>
    
    
    <footer>
      <section id="footer">
        <div class="wrapper">
          <div class="text_right">
            <p>© 2025 Spruce Emmanuelp>
          div>
        div>
      section>
    footer>
    

    Breakdown of the HTML:

    • Contact section: This section includes a welcoming text that invites users to reach out, followed by a contact form with fields for the user’s name, email, and a message.

    • Footer: The footer includes copyright information, ensuring that visitors know who owns the content. It’s placed at the bottom of the page and gives a professional touch to the site.

    CSS for the Contact Section and Footer

    Now let's style the Contact Section and Footer to make them both user-friendly and visually appealing.

    Contact section styling:

    /* contact section */
    .contact_form {
      display: flex;
      flex-direction: column;
      gap: 20px;
    }
    
    .form_group {
      display: flex;
      flex-direction: column;
    }
    
    label {
      margin-bottom: 5px;
      font-weight: bold;
    }
    
    input[type="text"],
    input[type="email"],
    textarea {
      width: 100%;
      padding: 10px;
      border: 1px solid #ccc;
      border-radius: 4px;
      box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
      margin-bottom: 10px;
    }
    
    textarea {
      resize: vertical;
      height: 150px;
    }
    
    button[type="submit"] {
      padding: 10px 20px;
      border: none;
      border-radius: 4px;
      background-color: #007bff;
      color: white;
      cursor: pointer;
    }
    
    button[type="submit"]:hover {
      background-color: #0056b3;
    }
    

    Footer section styling:

    /* footer section */
    section#footer {
      padding: var(--space-l) var(--space-l);
    }
    

    Wrapping Up

    And there you have it! You've just built a simple portfolio website using only HTML and CSS. You've learned how to structure your project, apply a CSS reset, and create responsive typography and layout systems that scale beautifully across different screen sizes.

    By following this guide, you now have a portfolio that you can be proud of and share with friends, family, or potential employers. The beauty of this project is that you can continue to expand and personalize it by adding new sections, showcasing more of your work, or experimenting with JavaScript to add interactive elements.

    If you're interested in exploring more advanced features or learning how to host your portfolio online, check out my tutorial on How to Host Your Project on GitHub – Explained With Examples.

    If you have any questions, feel free to find me on Twitter at @sprucekhalifa, and don’t forget to follow me for more tips and updates. Happy coding!