In this tutorial, I will teach you how to create a Todo List app using HTML5, CSS3, and JavaScript. The complete source code of this JavaScript to-do list app is given below.
You can download the full source code (including images) of this JavaScript todo list app at the end of this article.
<!DOCTYPE html> <html lang="en" dir="ltr"> <head> <meta charset="utf-8"> <title>Todo List App in JavaScript</title> <link rel="stylesheet" href="style.css"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <!-- Iconscout Link For Icons --> <link rel="stylesheet" href="https://unicons.iconscout.com/release/v4.0.0/css/line.css"> </head> <body> <div class="wrapper"> <div class="task-input"> <img src="bars-icon.svg" alt="icon"> <input type="text" placeholder="Add a new task"> </div> <div class="controls"> <div class="filters"> <span class="active" id="all">All</span> <span id="pending">Pending</span> <span id="completed">Completed</span> </div> <button class="clear-btn">Clear All</button> </div> <ul class="task-box"></ul> </div> <script src="script.js"></script> </body> </html>
const taskInput = document.querySelector(".task-input input"), filters = document.querySelectorAll(".filters span"), clearAll = document.querySelector(".clear-btn"), taskBox = document.querySelector(".task-box"); let editId, isEditTask = false, todos = JSON.parse(localStorage.getItem("todo-list")); filters.forEach(btn => { btn.addEventListener("click", () => { document.querySelector("span.active").classList.remove("active"); btn.classList.add("active"); showTodo(btn.id); }); }); function showTodo(filter) { let liTag = ""; if(todos) { todos.forEach((todo, id) => { let completed = todo.status == "completed" ? "checked" : ""; if(filter == todo.status || filter == "all") { liTag += `<li class="task"> <label for="${id}"> <input >
/* Import Google Font - Poppins */@import url('https://fonts.googleapis.com/css2?family=Poppins:wght@400;500;600;700&display=swap'); *{ margin: 0; padding: 0; box-sizing: border-box; font-family: 'Poppins', sans-serif; } body{ width: 100%; height: 100vh; overflow: hidden; background: linear-gradient(135deg, #4AB1FF, #2D5CFE); } ::selection{ color: #fff; background: #3C87FF; } .wrapper{ max-width: 405px; padding: 28px 0 30px; margin: 137px auto; background: #fff; border-radius: 7px; box-shadow: 0 10px 30px rgba(0,0,0,0.1); } .task-input{ height: 52px; padding: 0 25px; position: relative; } .task-input img{ top: 50%; position: absolute; transform: translate(17px, -50%); } .task-input input{ height: 100%; width: 100%; outline: none; font-size: 18px; border-radius: 5px; padding: 0 20px 0 53px; border: 1px solid #999; } .task-input input:focus, .task-input input.active{ padding-left: 52px; border: 2px solid #3C87FF; } .task-input input::placeholder{ color: #bfbfbf; } .controls, li{ display: flex; align-items: center; justify-content: space-between; } .controls{ padding: 18px 25px; border-bottom: 1px solid #ccc; } .filters span{ margin: 0 8px; font-size: 17px; color: #444444; cursor: pointer; } .filters span:first-child{ margin-left: 0; } .filters span.active{ color: #3C87FF; } .controls .clear-btn{ border: none; opacity: 0.6; outline: none; color: #fff; cursor: pointer; font-size: 13px; padding: 7px 13px; border-radius: 4px; letter-spacing: 0.3px; pointer-events: none; transition: transform 0.25s ease; background: linear-gradient(135deg, #1798fb 0%, #2D5CFE 100%); } .clear-btn.active{ opacity: 0.9; pointer-events: auto; } .clear-btn:active{ transform: scale(0.93); } .task-box{ margin-top: 20px; margin-right: 5px; padding: 0 20px 10px 25px; } .task-box.overflow{ overflow-y: auto; max-height: 300px; } .task-box::-webkit-scrollbar{ width: 5px; } .task-box::-webkit-scrollbar-track{ background: #f1f1f1; border-radius: 25px; } .task-box::-webkit-scrollbar-thumb{ background: #e6e6e6; border-radius: 25px; } .task-box .task{ list-style: none; font-size: 17px; margin-bottom: 18px; padding-bottom: 16px; align-items: flex-start; border-bottom: 1px solid #ccc; } .task-box .task:last-child{ margin-bottom: 0; border-bottom: 0; padding-bottom: 0; } .task-box .task label{ display: flex; align-items: flex-start; } .task label input{ margin-top: 7px; accent-color: #3C87FF; } .task label p{ user-select: none; margin-left: 12px; word-wrap: break-word; } .task label p.checked{ text-decoration: line-through; } .task-box .settings{ position: relative; } .settings :where(i, li){ cursor: pointer; } .settings .task-menu{ z-index: 10; right: -5px; bottom: -65px; padding: 5px 0; background: #fff; position: absolute; border-radius: 4px; transform: scale(0); transform-origin: top right; box-shadow: 0 0 6px rgba(0,0,0,0.15); transition: transform 0.2s ease; } .task-box .task:last-child .task-menu{ bottom: 0; transform-origin: bottom right; } .task-box .task:first-child .task-menu{ bottom: -65px; transform-origin: top right; } .task-menu.show{ transform: scale(1); } .task-menu li{ height: 25px; font-size: 16px; margin-bottom: 2px; padding: 17px 15px; cursor: pointer; justify-content: flex-start; } .task-menu li:last-child{ margin-bottom: 0; } .settings li:hover{ background: #f5f5f5; } .settings li i{ padding-right: 8px; } @media (max-width: 400px) { body{ padding: 0 10px; } .wrapper { padding: 20px 0; } .filters span{ margin: 0 5px; } .task-input{ padding: 0 20px; } .controls{ padding: 18px 20px; } .task-box{ margin-top: 20px; margin-right: 5px; padding: 0 15px 10px 20px; } .task label input{ margin-top: 4px; } }
MiniMax-M1 is a new open-weight large language model (456 B parameters, ~46 B active) built with hybrid…
Managing Git hooks manually can quickly become tedious and error-prone—especially in fast-moving JavaScript or Node.js…
Git hooks help teams enforce code quality by automating checks at key stages like commits…
Choosing the right Git hooks manager directly impacts code quality, developer experience, and CI/CD performance.…
We evaluated the performance of Llama 3.1 vs GPT-4 models on over 150 benchmark datasets…
The manufacturing industry is undergoing a significant transformation with the advent of Industrial IoT Solutions.…