Header Image
Thumbnail Image

Modern Interactive To-Do List with HTML, CSS, and JavaScript

By Cristian Quiñones, Published on January 21st 2025 | 9 mins, 1717 words

Today we will be creating a Modern Interactive To-Do List with just HTML, CSS, and JavaScript. This tutorial will give you a basic understanding on how to create a clean and user-friendly to-do list application with interactive functionalities like adding task, filtering, and task deletion. 


after finishing this project, you will have a complete functional and  stylish  to-do list that you can customize to fit your own projects. Let's get into the code!




 Step 1: Setting Up the Front End Layout


Let's start by creating a new file called index.hmtl then follow this command:



<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <link rel="stylesheet" href="style.css">
    <title>Modern Interactive Todo List</title>
    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css" />
</head>
<body>
  <main class="todo-container">
    <header>
      <h1><i class="fas fa-check-circle"></i> To Do List</h1>
    </header>
    <section class="input-section">
      <input type="text" id="task-input" placeholder="Add a new task..." aria-label="Add a new task">
      <button id="add-task-btn"><i class="fas fa-plus"></i> Add</button>
    </section>
    <nav class="filters" aria-label="Task Filters">
      <button class="filter-btn active" data-filter="all"><i class="fas fa-list"></i> All</button>
      <button class="filter-btn" data-filter="active"><i class="fas fa-tasks"></i> Active</button>
      <button class="filter-btn" data-filter="completed"><i class="fas fa-check"></i> Completed</button>
    </nav>
    <ul class="task-list" aria-live="polite">
    </ul>
  </main>

  <script src="https://cdnjs.cloudflare.com/ajax/libs/Sortable/1.15.0/Sortable.min.js"></script>

    <script src="main.js"></script>
</body>
</html>




Step 2: Styling the To-Do List


Let's put some styling in our front end structure to make it more visually appealing. Create a new file called style.css and follow this command:



/* style.css */

/* CSS Variables for Efficient Color Management */
:root {
  --background-color: #1b1b2f;
  --container-bg: #2c2c54;
  --accent-color: #1de9b6;
  --button-gradient: linear-gradient(45deg, #00c6ff, #0072ff);
  --completed-task-color: #6a6a6a;
  --text-color: #ffffff;
  --input-bg: #3d3d5c;
  --delete-color: #ff6b6b;
  --delete-hover-color: #ff4757;
  --filter-active-color: #1de9b6;
  --border-radius: 10px;
}

* {
  box-sizing: border-box;
}

body {
  display: flex;
  align-items: center;
  justify-content: center;
  min-height: 100vh;
  background-color: var(--background-color);
  margin: 0;
  font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
  color: var(--text-color);
  overflow: hidden;
}

.todo-container {
  background: var(--container-bg);
  padding: 40px;
  border-radius: 15px;
  box-shadow: 0 0 20px rgba(0, 0, 0, 0.5);
  position: relative;
  width: 400px;
  max-width: 90%;
  overflow: hidden;
  animation: fadeIn 1s ease-out;
}

.todo-container::before {
  content: '';
  position: absolute;
  top: -30px;
  left: -30px;
  right: -30px;
  bottom: -30px;
  background: linear-gradient(45deg, var(--accent-color), #6a00f4, #ff4081, var(--accent-color));
  background-size: 400% 400%;
  border-radius: 20px;
  filter: blur(50px);
  animation: pulseGlow 6s ease-in-out infinite;
  z-index: -1;
}

header h1 {
  text-align: center;
  margin-bottom: 20px;
  font-size: 2em;
  position: relative;
  z-index: 1;
  display: flex;
  align-items: center;
  justify-content: center;
}

header h1 i {
  margin-right: 10px;
  color: var(--accent-color);
  animation: rotateIcon 4s linear infinite;
}

@keyframes rotateIcon {
  0% { transform: rotate(0deg); }
  100% { transform: rotate(360deg); }
}

.input-section {
  display: flex;
  margin-bottom: 20px;
}

#task-input {
  flex: 1;
  padding: 12px 20px;
  border: none;
  border-radius: 50px;
  outline: none;
  font-size: 1em;
  background: var(--input-bg);
  color: var(--text-color);
  transition: box-shadow 0.3s ease, background 0.3s ease;
}

#task-input::placeholder {
  color: #ccc;
}

#task-input:focus {
  box-shadow: 0 0 10px var(--accent-color);
  background: #4e4e7a;
}

#add-task-btn {
  margin-left: 10px;
  padding: 12px 20px;
  background: var(--button-gradient);
  border: none;
  border-radius: 50px;
  color: var(--text-color);
  font-weight: bold;
  cursor: pointer;
  position: relative;
  overflow: hidden;
  transition: transform 0.3s ease, box-shadow 0.3s ease, background 0.3s ease;
  display: flex;
  align-items: center;
  justify-content: center;
}

#add-task-btn i {
  margin-right: 8px;
}

#add-task-btn::before {
  content: '';
  position: absolute;
  top: 50%;
  left: 50%;
  width: 300%;
  height: 300%;
  background: rgba(255, 255, 255, 0.2);
  border-radius: 50%;
  transform: translate(-50%, -50%) scale(0);
  transition: transform 0.5s ease;
}

#add-task-btn:hover::before {
  transform: translate(-50%, -50%) scale(1);
}

#add-task-btn:hover {
  transform: scale(1.05);
  box-shadow: 0px 4px 20px rgba(0, 255, 255, 0.4);
  background: linear-gradient(45deg, #ff4081, var(--accent-color));
}

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

.filter-btn {
  background: none;
  border: none;
  color: #ccc;
  padding: 10px 15px;
  cursor: pointer;
  position: relative;
  transition: color 0.3s ease;
  display: flex;
  align-items: center;
  font-size: 0.9em;
}

.filter-btn i {
  margin-right: 5px;
}

.filter-btn.active,
.filter-btn:hover {
  color: var(--text-color);
}

.filter-btn::after {
  content: '';
  position: absolute;
  bottom: 0;
  left: 50%;
  width: 0;
  height: 2px;
  background: var(--accent-color);
  transition: width 0.3s ease, left 0.3s ease;
}

.filter-btn.active::after,
.filter-btn:hover::after {
  width: 100%;
  left: 0;
}

.task-list {
  list-style: none;
  padding: 0;
  max-height: 300px;
  overflow-y: auto;
  position: relative;
}

.task-item {
  background: var(--input-bg);
  padding: 15px 20px;
  border-radius: var(--border-radius);
  margin-bottom: 10px;
  display: flex;
  align-items: center;
  justify-content: space-between;
  transition: transform 0.3s ease, opacity 0.3s ease, background 0.3s ease;
  position: relative;
}

.task-item.completed {
  background: var(--completed-task-color);
  text-decoration: line-through;
  opacity: 0.6;
}

.task-item .task-text {
  flex: 1;
  margin-left: 15px;
  word-break: break-word;
}

.task-item .task-text.completed {
  color: #b3b3b3;
}

.delete-btn {
  background: rgba(255, 255, 255, 0.1); /* Slightly transparent background */
  border: none;
  color: var(--delete-color);
  cursor: pointer;
  font-size: 1.4em; /* Increased icon size */
  transition: color 0.3s ease, background 0.3s ease;
  border-radius: 50%;
  width: 35px;
  height: 35px;
  display: flex;
  align-items: center;
  justify-content: center;
  position: relative;
}

.delete-btn i {
  pointer-events: none;
}

.delete-btn:hover {
  color: var(--delete-hover-color);
  background: rgba(255, 255, 255, 0.2); /* Lighter background on hover */
  box-shadow: 0 0 10px var(--delete-hover-color); /* Glow effect */
}

/* Adding a glow effect on hover */
.task-item:hover .delete-btn {
  box-shadow: 0 0 10px var(--accent-color);
}

/* Additional button animations */
.delete-btn::after {
  content: '';
  position: absolute;
  top: 50%;
  left: 50%;
  width: 0;
  height: 0;
  background: var(--delete-hover-color);
  border-radius: 50%;
  transform: translate(-50%, -50%);
  opacity: 0;
  transition: all 0.3s ease;
}

.delete-btn:hover::after {
  width: 100%;
  height: 100%;
  opacity: 0.2;
}

@keyframes pulseGlow {
  0% {
    background-position: 0% 0%;
  }
  50% {
    background-position: 100% 100%;
    opacity: 1;
  }
  100% {
    background-position: 0% 0%;
    opacity: 0.8;
  }
}

@keyframes fadeIn {
  from {
    opacity: 0;
    transform: scale(0.95);
  }
  to {
    opacity: 1;
    transform: scale(1);
  }
}

/* Custom Scrollbar */
.task-list::-webkit-scrollbar {
  width: 8px;
}

.task-list::-webkit-scrollbar-track {
  background: #2c2c54;
}

.task-list::-webkit-scrollbar-thumb {
  background: var(--accent-color);
  border-radius: 4px;
}

.task-list::-webkit-scrollbar-thumb:hover {
  background: #1de9b6;
}

/* Fade-out Animation for Deletion */
.fade-out {
  opacity: 0;
  transform: translateX(100%);
  transition: all 0.5s ease;
}

/* Glow Effect on Tasks */
.task-item::before {
  content: '';
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  background: linear-gradient(45deg, var(--accent-color), #6a00f4, #ff4081, var(--accent-color));
  background-size: 400% 400%;
  border-radius: var(--border-radius);
  filter: blur(10px);
  opacity: 0;
  transition: opacity 0.3s ease;
  z-index: -1;
}

.task-item:hover::before {
  opacity: 0.7;
  animation: pulseGlow 6s ease-in-out infinite;
}

/* Styles for Notifications */
.notification {
  position: fixed;
  top: 20px;
  right: 20px;
  background: rgba(255, 255, 255, 0.1);
  color: var(--text-color);
  padding: 15px 20px;
  border-radius: var(--border-radius);
  backdrop-filter: blur(10px);
  box-shadow: 0 0 10px rgba(0, 0, 0, 0.5);
  z-index: 1000;
  opacity: 0;
  transform: translateY(-20px);
}

.notification-info {
  border-left: 5px solid var(--accent-color);
}

.notification-success {
  border-left: 5px solid #28a745;
}

.notification-error {
  border-left: 5px solid #dc3545;
}

.notification-warning {
  border-left: 5px solid #ffc107;
}

.notification.show {
  opacity: 1;
  transform: translateY(0);
  transition: all 0.5s ease;
}

/* Responsive Design */
@media (max-width: 500px) {
  .todo-container {
    padding: 20px;
    width: 90%;
  }

  header h1 {
    font-size: 1.5em;
  }

  #add-task-btn {
    padding: 10px 15px;
    font-size: 0.9em;
  }

  .filter-btn {
    padding: 8px 12px;
    font-size: 0.8em;
  }
}




 Step 3: Make our website more interactive with Javascript



Now lets add some JavaScript logic that drives the interactivity of the to-do list. Create a new file called main.js and follow this code:



// main.js

document.addEventListener('DOMContentLoaded', () => {
    const taskInput = document.getElementById('task-input');
    const addTaskBtn = document.getElementById('add-task-btn');
    const taskList = document.querySelector('.task-list');
    const filterButtons = document.querySelectorAll('.filter-btn');
    let currentFilter = 'all';
  
    // Load tasks from localStorage
    loadTasks();
  
    // Add a new task
    addTaskBtn.addEventListener('click', () => {
      const taskText = taskInput.value.trim();
      if (taskText !== '') {
        addTask(taskText);
        taskInput.value = '';
        saveTasks();
        showNotification('Task added successfully!', 'success');
      }
    });
  
    // Add a task by pressing Enter
    taskInput.addEventListener('keypress', (e) => {
      if (e.key === 'Enter') {
        addTaskBtn.click();
      }
    });
  
    // Function to add a task
    function addTask(text, completed = false) {
      const li = document.createElement('li');
      li.classList.add('task-item');
      if (completed) li.classList.add('completed');
  
      const checkbox = document.createElement('input');
      checkbox.type = 'checkbox';
      checkbox.checked = completed;
      checkbox.setAttribute('aria-label', 'Mark as completed');
      checkbox.addEventListener('change', () => {
        li.classList.toggle('completed');
        const taskTextSpan = li.querySelector('.task-text');
        taskTextSpan.classList.toggle('completed');
        saveTasks();
        applyFilter();
      });
  
      const span = document.createElement('span');
      span.classList.add('task-text');
      span.textContent = text;
      if (completed) span.classList.add('completed');
  
      const deleteBtn = document.createElement('button');
      deleteBtn.classList.add('delete-btn');
      deleteBtn.innerHTML = '<i class="fas fa-trash"></i>';
      deleteBtn.setAttribute('aria-label', 'Delete task');
      deleteBtn.addEventListener('click', () => {
        if (confirm('Are you sure you want to delete this task?')) {
          li.classList.add('fade-out');
          li.addEventListener('transitionend', () => {
            li.remove();
            saveTasks();
            showNotification('Task deleted successfully!', 'success');
          });
        }
      });
  
      li.appendChild(checkbox);
      li.appendChild(span);
      li.appendChild(deleteBtn);
      taskList.appendChild(li);
  
      // Add animation
      li.style.opacity = 0;
      li.style.transform = 'translateY(-20px)';
      setTimeout(() => {
        li.style.transition = 'all 0.5s ease';
        li.style.opacity = 1;
        li.style.transform = 'translateY(0)';
      }, 10);
    }
  
    // Filter tasks
    filterButtons.forEach(btn => {
      btn.addEventListener('click', () => {
        filterButtons.forEach(b => b.classList.remove('active'));
        btn.classList.add('active');
        currentFilter = btn.getAttribute('data-filter');
        applyFilter();
      });
    });
  
    function applyFilter() {
      const tasks = document.querySelectorAll('.task-item');
      tasks.forEach(task => {
        if (currentFilter === 'all') {
          task.style.display = 'flex';
        } else if (currentFilter === 'active') {
          task.style.display = task.classList.contains('completed') ? 'none' : 'flex';
        } else if (currentFilter === 'completed') {
          task.style.display = task.classList.contains('completed') ? 'flex' : 'none';
        }
      });
    }
  
    // Save tasks to localStorage
    function saveTasks() {
      const tasks = [];
      const taskItems = document.querySelectorAll('.task-item');
      taskItems.forEach(item => {
        const taskText = item.querySelector('.task-text').textContent;
        const completed = item.classList.contains('completed');
        tasks.push({ text: taskText, completed });
      });
      localStorage.setItem('tasks', JSON.stringify(tasks));
    }
  
    // Load tasks from localStorage
    function loadTasks() {
      const tasks = JSON.parse(localStorage.getItem('tasks')) || [];
      tasks.forEach(task => addTask(task.text, task.completed));
    }
  
    // Initialize SortableJS for drag-and-drop functionality
    if (typeof Sortable !== 'undefined') {
      const sortable = Sortable.create(taskList, {
        animation: 150,
        onEnd: function () {
          saveTasks();
        }
      });
    }
  
    // Function to display notifications
    function showNotification(message, type = 'info') {
      const notification = document.createElement('div');
      notification.classList.add('notification', `notification-${type}`);
      notification.textContent = message;
  
      document.body.appendChild(notification);
  
      // Trigger reflow to enable transition
      void notification.offsetWidth;
  
      // Add show class for animation
      notification.classList.add('show');
  
      // Remove notification after 3 seconds
      setTimeout(() => {
        notification.classList.remove('show');
        notification.addEventListener('transitionend', () => {
          notification.remove();
        });
      }, 3000);
    }
  });
  




Testing the To-Do List Application

Once you’ve written your HTML, CSS, and JavaScript, it’s time to test the functionality of our to-do list. Now open your index.html file in the browser.


moderntodolist.png 33.81 KB



Congratulations! You’ve just built a Modern Interactive To-Do List with HTML, CSS, and JavaScript. This simple to-do list application can be enhance further with additional features like task prioritization, date tracking, or saving tasks to local storage.

Feel free to experiment with the code, tweak the styles, and add more functionality to suit your needs. Happy coding!










Discussion (0)

Log in to comment!

No comments yet!