forked from stefany-newman/accessible-to-do-list
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscript.js
112 lines (103 loc) · 3.11 KB
/
script.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
const to_do_input = document.querySelector('#todo');
const list = document.querySelector('#list');
const to_do_form = document.querySelector('#to_do_form');
const sc_feedback = document.querySelector('#sc_feedback');
const heading = document.querySelector('#heading');
/*
Execute functions on submit of the form
*/
to_do_form.addEventListener('submit', function (event) {
event.preventDefault();
let task = to_do_input.value;
addTaskToDOM(task);
removeValue(to_do_input);
screenReaderFeedback(task);
});
/*
We are using Event Bubbling to listen to when the <ul> element
that keeps the tasks is clicked. When it is clicked, if the click
originates on the delete button, then we execute the delete
task functionality.
*/
list.addEventListener('click', function (event) {
if (hasClassName(event.target, 'delete_task')) {
const li = event.target.closest('li'); // Find the parent <li> of the clicked delete button
const taskName = event.target.previousElementSibling.textContent; // Get the task text
deleteTask(li);
moveFocus(heading);
screenReaderFeedback(taskName, 'removed');
}
});
function deleteTask(theTarget) {
list.removeChild(theTarget);
}
/*
We generate a new <li>, <input type="checkbox">, and <button>
each time a task is added to the DOM.
*/
function addTaskToDOM(task) {
let newID = generateID();
let taskItem = createElement('li', '', list, ['class', 'task']);
let theCheckbox = createElement('input', null, taskItem, [
'type',
'checkbox',
]);
let label = createElement('label', task, taskItem, ['for', newID]);
theCheckbox.setAttribute('id', newID);
let deleteButton = createElement('button', 'Delete Task', taskItem, [
'class',
'delete_task',
]);
}
function createElement(tagName, textNode, parent, attribute = null) {
let node = document.createElement(tagName);
if (textNode != null) {
let customTextNode = document.createTextNode(textNode);
node.appendChild(customTextNode);
}
if (attribute != null) {
node.setAttribute(attribute[0], attribute[1]);
}
parent.appendChild(node);
return node;
}
function removeValue(input) {
input.value = '';
}
/*
Sighted users are able to see when a task is added, but screen reader
users are more likely to be blind, so we want to make the
screen reader voice the actions.
We use the #sc_feedback element in the DOM to populate
the text which would be voiced
*/
function screenReaderFeedback(task, feedback = 'added') {
sc_feedback.textContent = `${task} ${feedback}.`;
}
/*
We want the checkbox to go before the label,
so we need to connect the labels to the inputs
with IDs.
*/
function generateID() {
let idPrefix = 'task_num_';
let tasks = document.querySelectorAll('#list > li');
if (tasks.length == 0) {
return `${idPrefix}0`;
}
return idPrefix + tasks.length;
}
function moveFocus(element) {
element.focus();
}
function hasClassName(element, className) {
/*
We are using classList.contains() instead
of element.className because an element may
have multiple classes
*/
if (element.classList.contains(className)) {
return true;
}
return false;
}