Lesson 12
Which of the following will output 42
?
console.log(a); // [1]
function foo() {
console.log(a); // [2]
let a = 42;
console.log(a); // [3]
}
console.log(a); // [4]
Which of the following will output 42
?
console.log(a); // [1] ERROR
function foo() {
console.log(a); // [2] ERROR
let a = 42;
console.log(a); // [3] 42
}
console.log(a); // [4] ERROR
function fun1() {
let veryImportantVariable = 42;
}
function fun2() {
// Question: How do I access veryImportantVariable????
}
// move the variable outside the function
let veryImportantVariable = 42;
function fun1() {
}
function fun2() {
}
What's the difference?
let a = 42;
let b = 43;
a = 42;
b = 43;
What's the difference?
let a = 42;
let b = 43;
a = 42;
b = 43;
The last one creates global variables. Those will be visible to everyone, your colleagues, third-party libraries, etc. etc. DO NOT EVER DO THAT. Your colleagues will hate you, your boss will shout at you, your CEO will point at you and your teacher will cry.
What's the difference?
function onMyButtonClick() {
let myElement = document.getElementById("input");
myElement.value = 42;
}
let myElement = document.getElementById("input");
function onMyButtonClick() {
myElement.value = 42;
}
Fist one asks for the input
element every time the button is clicked, last one only once when the page loads.
- Create the usual empty HTML and JavaScript file
- Remember to load your JavaScript file in the body using a
<script src=""></script>
tag - Add one
<ul id="todoList"></ul>
to your HTML page - In your JavaScript file, add an array of things you need to do, for example:
let todoItems = [ "wash dishes", "learn JavaScript", "do sport" ];
- Add all the
todoItems
from your JavaScript Array to your<ul>
using the DOM API
let todoListElement = document.getElementById("todoList");
let todoItems = [ "wash dishes", "learn JavaScript", "do sport" ];
for (let i = 0; i < todoItems.length; i++) {
let todoElement = document.createElement("li");
todoElement.textContent = todoItems[i];
todoListElement.appendChild(todoElement);
}
- We already learned that HTML Elements have a
onclick
property. - We know that this property must be a function.
- We know that we can declare functions like:
function foo() {
console.log("foo was called");
}
- However, variables can be functions, too:
let foo = function() {
console.log("foo was called");
}
- What this means: Create a new variable called
foo
and initialize it with a function
- Variables containing a function can be used exactly like variables containing a
number
,array
orobject
:
let foo = function() {
console.log("foo was called");
}
let bar = foo; // now foo and bar point to the same function
bar(); // prints "foo was called"
- The function can access all variables visible to it (technical term: "Closure"):
let x = 42;
let foo = function() {
console.log("foo was called with x: " + x);
}
let bar = foo; // now foo and bar point to the same function
bar(); // prints "foo was called with x: 42"
- So if we want to set an
onclick
handler:
let myClickFunction = function() { /* some code */ }
document.body.onclick = myClickFunction;
- Or directly:
document.body.onclick = function() { /* some code */ }
- Can you add a onclick handler to each of your
<li>
elements that changes the text to strike through (e.g.wash dishes)? - Hint: Use the
textDecoration
property:
todoElement.style.textDecoration = "line-through";
let todoListElement = document.getElementById("todoList");
let todoItems = [ "wash dishes", "learn JavaScript", "do sport" ];
for (let i = 0; i < todoItems.length; i++) {
let todoElement = document.createElement("li");
todoElement.textContent = todoItems[i];
todoElement.onclick = function() {
todoElement.style.textDecoration = "line-through";
}
todoListElement.appendChild(todoElement);
}
- Can you remove the strike-through if a strike-through item is clicked?
todoElement.onclick = function() {
if (todoElement.style.textDecoration === "line-through") {
todoElement.style.textDecoration = "";
} else {
todoElement.style.textDecoration = "line-through";
}
}
- HTML Elements are hierarchical:
<body>
<ul>
<li>Wash dishes</li>
</ul>
</body>
- Every HTML Element has zero or more children
- Every HTML Element has a children property:
let items = todoListElement.children;
- The property returns an array of all direct children
- We can use a
for
loop to go through all children:
let items = todoListElement.children;
for (let i = 0; i < items.length; i++) {
console.log("item: " + items[i].textContent);
}
- Can you add a button "Mark all done" that strikes through all your TODO items?
- Hint: Use the
children
property of your<ul>
element. - BONUS: Add an
<input>
field and a<button>
to dynamically add new TODO items to your<ul>