Building a Goal Tracker with HTML, CSS, and JavaScript

Goal tracking is an essential aspect of personal development and productivity. By visualizing your progress and breaking down tasks into manageable steps, you can stay motivated and focused. In this tutorial, we will create a simple yet effective Goal Tracker application using HTML, CSS, and JavaScript. This application will allow users to set goals, track their progress, and see their achievements over time.

In this tutorial, we will create a simple yet effective Goal Tracker application using HTML, CSS, and JavaScript. This application will allow users to set goals, track their progress, and see their achievements over time. Let’s dive into the code and see how it all comes together!

Introduction

Goal tracking is an essential aspect of personal development and productivity. By visualizing your progress and breaking down tasks into manageable steps, you can stay motivated and focused. This project will help you build a Goal Tracker app that you can customize and expand as needed.

Step 1: Setting Up the HTML Structure

First, let’s create the basic structure of our Goal Tracker. We will need a form to add new goals, tabs to switch between active and completed goals, and sections to display these goals

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Goal Tracker</title>
<style>
/* Add CSS styles here */
</style>
</head>
<body>
<div class="container">
<h1>Goal Tracker</h1>
<form id="goalForm">
<label for="goalName">Goal Name:</label>
<input type="text" id="goalName" required>
<label for="goalDuration">Duration (days):</label>
<input type="number" id="goalDuration" required>
<button type="submit">Create Goal</button>
</form>
<div class="tabs">
<button class="tab" onclick="showTab('active')">Active Goals</button>
<button class="tab" onclick="showTab('completed')">Completed Goals</button>
</div>
<div id="activeGoals" class="goalSection"></div>
<div id="completedGoals" class="goalSection" style="display:none;"></div>
<div id="pagination"></div>
</div>
<script>
/* Add JavaScript code here */
</script>
</body>
</html>


Step 2: Styling the Application

Next, we add some CSS to style our application. We will ensure the layout is clean and user-friendly.

body {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
background-color: #eef2f3;
margin: 0;
padding: 0;
}

.container {
width: 60%;
margin: 20px auto;
padding: 20px;
background: #fff;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
border-radius: 8px;
}

h1 {
text-align: center;
color: #333;
}

form {
display: flex;
flex-direction: column;
margin-bottom: 20px;
}

label, input, button {
margin-bottom: 10px;
}

input, button {
padding: 10px;
font-size: 1em;
}

button {
background-color: #007bff;
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
}

button:hover {
background-color: #0056b3;
}

.tabs {
display: flex;
justify-content: center;
margin-bottom: 20px;
}

.tab {
padding: 10px 20px;
background-color: #007bff;
color: white;
border: none;
cursor: pointer;
margin: 0 5px;
border-radius: 4px;
}

.tab:hover, .tab.active {
background-color: #0056b3;
}

.goalSection {
display: flex;
flex-direction: column;
gap: 15px;
}

.goal {
padding: 15px;
background: #f8f9fa;
border-radius: 8px;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
}

.goal h2 {
margin-top: 0;
}

.days {
display: flex;
flex-wrap: wrap;
gap: 5px;
}

.day {
padding: 5px 10px;
background: #ddd;
border-radius: 4px;
}

.day.completed {
background: #28a745;
color: white;
}

.checkbox {
margin-right: 10px;
}

#pagination {
display: flex;
justify-content: center;
margin-top: 20px;
}

.pagination-button {
padding: 10px;
margin: 0 5px;
background: #007bff;
color: white;
border: none;
cursor: pointer;
border-radius: 4px;
}

.pagination-button.disabled {
background: #ccc;
cursor: not-allowed;
}




Step 3: Adding Functionality with JavaScript

Now, let’s add JavaScript to handle goal creation, goal tracking, and pagination. The script will manage our goals in local storage, render goals dynamically, and allow users to mark days as completed.

document.addEventListener('DOMContentLoaded', (event) => {
const goalForm = document.getElementById('goalForm');
const activeGoalsContainer = document.getElementById('activeGoals');
const completedGoalsContainer = document.getElementById('completedGoals');
const paginationContainer = document.getElementById('pagination');
const goals = JSON.parse(localStorage.getItem('goals')) || [];
let currentPage = 1;
const goalsPerPage = 5;

const saveGoals = () => {
localStorage.setItem('goals', JSON.stringify(goals));
};

const renderPagination = () => {
paginationContainer.innerHTML = '';
const totalPages = Math.ceil(goals.length / goalsPerPage);

for (let i = 1; i <= totalPages; i++) {
const button = document.createElement('button');
button.classList.add('pagination-button');
if (i === currentPage) {
button.classList.add('active');
}
button.innerText = i;
button.addEventListener('click', () => {
currentPage = i;
renderGoals();
});
paginationContainer.appendChild(button);
}
};

const renderGoals = () => {
activeGoalsContainer.innerHTML = '';
completedGoalsContainer.innerHTML = '';
const start = (currentPage - 1) * goalsPerPage;
const end = start + goalsPerPage;
const currentGoals = goals.slice(start, end);

currentGoals.forEach((goal, index) => {
const goalDiv = document.createElement('div');
goalDiv.classList.add('goal');

const goalTitle = document.createElement('h2');
goalTitle.innerText = goal.name;
goalDiv.appendChild(goalTitle);

const daysDiv = document.createElement('div');
daysDiv.classList.add('days');

goal.completedDays.forEach((completed, i) => {
const dayDiv = document.createElement('div');
dayDiv.classList.add('day');
if (completed) {
dayDiv.classList.add('completed');
}

const checkbox = document.createElement('input');
checkbox.type = 'checkbox';
checkbox.classList.add('checkbox');
checkbox.checked = completed;
checkbox.addEventListener('change', () => {
goal.completedDays[i] = checkbox.checked;
goal.points = goal.completedDays.filter(Boolean).length;
saveGoals();
renderGoals();
});

const dayLabel = document.createElement('span');
dayLabel.innerText = `Day ${i + 1}`;

dayDiv.appendChild(checkbox);
dayDiv.appendChild(dayLabel);
daysDiv.appendChild(dayDiv);
});

const points = document.createElement('p');
points.innerText = `Points: ${goal.points}`;
goalDiv.appendChild(daysDiv);
goalDiv.appendChild(points);

if (goal.completedDays.every(Boolean)) {
completedGoalsContainer.appendChild(goalDiv);
} else {
activeGoalsContainer.appendChild(goalDiv);
}
});

renderPagination();
};

goalForm.addEventListener('submit', (event) => {
event.preventDefault();
const goalName = document.getElementById('goalName').value;
const goalDuration = parseInt(document.getElementById('goalDuration').value);
const newGoal = {
name: goalName,
duration: goalDuration,
completedDays: Array(goalDuration).fill(false),
points: 0
};

goals.push(newGoal);
saveGoals();
renderGoals();
goalForm.reset();
});

window.showTab = (tabName) => {
if (tabName === 'active') {
activeGoalsContainer.style.display = 'flex';
completedGoalsContainer.style.display = 'none';
} else if (tabName === 'completed') {
activeGoalsContainer.style.display = 'none';
completedGoalsContainer.style.display = 'flex';
}
};

renderGoals();
});


Conclusion

You can explore the full codebase of the Goal Tracker project on GitHub by visiting the repository here. This repository contains all the HTML, CSS, and JavaScript files that power the application, allowing you to delve into the code and understand how the Goal Tracker works. Additionally, you can experience the live version of the app here, where you can interact with its features and see how goals are tracked and managed in real-time. Whether you’re interested in studying the code or trying out the application yourself, both the GitHub repository and the live link provide valuable resources for learning and exploration.

With these steps, you now have a fully functional Goal Tracker application. This project demonstrates how to handle form inputs, manage state with local storage, dynamically render HTML, and implement pagination. Feel free to expand this project with additional features such as goal editing, deletion, or progress charts.

I hope you enjoyed this tutorial and found it helpful for your own projects. Happy coding!

Ajink Gupta
Ajink Gupta

Ajink Gupta is a software developer from Dombivli, Maharashtra, India. He has expertise in a variety of technologies including web development, mobile app development, and blockchain. He works with languages and frameworks like JavaScript, Python, Flutter, React, and Django.

Ajink Gupta is also active on several platforms where he shares his work and engages with the community. You can find his projects and contributions on GitHub and follow his tutorials and updates on his YouTube channel​ . He also has a personal website where he showcases his portfolio and ongoing projects at ajinkgupta.vercel.app

Articles: 61

jsDelivr CDN plugin by Nextgenthemes

These are the assets loaded from jsDelivr CDN. Do not worry about old WP versions in the URLs, this is simply because the files were not modified. A sha384 hash check is used so you can be 100% sure the files loaded from jsDelivr are the exact same files that would be served from your server.


	

Level up your video embeds with ARVE or ARVE Pro