Step 8: marking items as done

Marking items as done

The next feature of our app will be marking items as done. To do this we need to rethink how we are representing our task items.

What we now call “a task” is a simple string in our code (a base value), and we cannot store other information there. We need to find a new way for representing our task, something that allows us to store other information together with the text of the task (e.g. whether the task is completed or not completed). What we need here is a compound value so we can group more than one value in a single entity. In JavaScript there are two compound values: Array (which we already know), and Object. Let’s use an Object.

Our task list will now become something like this:

8.1 – the new listItem with objects instead of strings

let listItems = [
  { text: 'Buy coffee',  completed: true  },
  { text: 'Buy milk',    completed: false },
  { text: 'Disco dance', completed: false }
];

Each task is now an Object with two properties: text (containing a String value) and completed (containing a Boolean value). These properties are accessible via the dot notation:

let item = { text: 'Example task', completed: true };
console.log(item.text); // will print 'Example task'
console.log(item.completed); // will print true

This is an important change in our app. We are changing the input data and this forces us to modify our logic accordingly – otherwise our app will no longer work.

Let’s modify our functions to be compatible with this new input format.

8.2 – the new renderItem() function

const renderItem = (item) => {
  const template = document.getElementById("item-template");
  let instance = document.importNode(template.content, true);
  instance.querySelector("span").innerText = item.text;
  return instance;
}

8.3 – the new createNew() function

const createNew = (event) => {
  event.preventDefault();

  const newItemElement = document.querySelector('#new-item');
  let newItemValue = newItemElement.value.trim();

  if (!newItemValue) return;

  listItems.push({ text: newItemValue, completed: false });
  newItemElement.value = '';

  updateList(listItems);
}

8.4 – the new removeItem() function

const removeItem = (event) => {
  const clickedItemText = event.target.previousElementSibling.innerHTML;

  listItems = listItems.filter((item) => clickedItemText != item.text);

  updateList(listItems);
}

With this important change, we are given the ability to store other information in our task!

Now we can implement the mark as done feature. The following is what we want to happen: when the user clicks on a incomplete task, we want to turn it into a completed task, and when he clicks on a completed task, we want it to return to the incomplete state.

A task can have two states: completed and not completed. This is a perfect fit for the Boolean value, and that’s the reason why we decided to use it. Another advantage of using a Boolean value is that it can be “inverted” easily, without checking its value first.

Let’s add an onclick attribute to our task <span> element inside our template:

<span onclick="toggleStatus(event)"></span>

...and create the corresponding function:

8.5 – the new toggleStatus() function

const toggleStatus = (event) => {
  const clickedItemText = event.target.innerHTML;

  listItems.forEach((item) => {
    if (clickedItemText == item.text) {
      item.completed = !item.completed;
    }
  });

  updateList(listItems);
}

This function will cycle through each item contained in listItems, and when it finds an element with matching text, it will invert its completed property.

We have modified the value of items in our listItems object but no change is visible yet. Let's modify our template by adding an attribute to our item:

8.6 – adding attribute

<li data-completed=false class="list-group-item">

And eventually, let's change this attribute by modifying the renderItem function so it reflects the status of the task!

8.7 – render the completeness status

const renderItem = (item) => {
  const template = document.getElementById("item-template");
  let instance = document.importNode(template.content, true);
  instance.querySelector("span").innerText = item.text;
  instance
    .querySelector(".list-group-item")
    .setAttribute("data-completed", item.completed);
  return instance;
}

What we just did:

  • we changed our input data from a list of string to a list of objects

  • we added a toggleStatus() function that inverts the completed property of the clicked task

  • we have rendered the status of the task

Last updated