- How to open console in browser – Ctrl+ shift + J or right click+ inspect or go to view-> developer ->console
alert("Hello World");
let jess = "Amazing";
if (jess === "Amazing") alert("JavaScript is fun");
let name = prompt("who are you?");
- High Level, Object Oriented, Multi Paradigm Programming Language
- High Level – we don’t have to worry about complex stuff like memory management
- Object Oriented – Based on Objects for storing most kinds of data
- Multi Paradigm – we can use different styles for programming
- Programming Language – Instruct Computer to do Things
- Core of website - HTML CSS JS
- HTML - content
- CSS - presentation
- JS - Programming Language
<script>
let js = 'amazing'; if(js === 'amazing') alert('JavaScript is FUN!");
</script>
console.log("Write here anything");
- In most cases, having JavaScript external is still recommended. But for certain cases, especially very small pieces of code, inlining them into the site’s HTML makes sense.
<script src="script.js"></script>
- As we know that HTML is loaded and executed line by line. So, when the browser encounters a
<script>
tag, it loads and executes the javascript code on the spot. This may slow down the page rendering speed and thus webpage will take more time to load.
- JavaScript uses reserved keyword var or let to declare a variable. A variable must have a unique name. You can assign a value to a variable using equal to (=) operator.
- camelCase - camelCase is useful in programming since element names cannot contain spaces. The camelCase naming convention makes compound names more readable.
- Variable names cannot contain spaces.
- Variable names must begin with a letter, an underscore (_) or a dollar sign ($).
- Variable names can only contain letters, numbers, underscores, or dollar signs.
- Variable names are case-sensitive.
- Certain words may not be used as variable names, because they have other meanings within JavaScript. Check out this complete list of the reserved words.
- All capital letter names must be used for constants.
- First letter capital must be used for OOP in Javascript.
- Number - Floating point numbers used for decimals and integers.
- String - Sequence of characrer used for text.
- Boolean - Logical type that can only be true or false used for taking decisions.
- Undefined - Value taken by a variable that is not yet defined(empty value).
- Null - Also means empty value.
- Symbol - Value that is unique and cannot be changed [Not useful for now].
- BigInt - Larger integers than the Number type can hold.
- Javascript has dynamic typing - We do not have to manually define the data type of the value stored in a variable. Insead, data types are determined automatically.
- Value has type, NOT variable.
// It is a comment
/*
It is multiline comment
*/
- The typeof operator returns the data type of its operand in the form of a string. The operand can be any object, function, or variable. Syntax: typeof operand. OR typeof (operand).
- A bug in JavaScript,
typeof null
is null but it shows object.
Var, Let, and Const – What's the Difference?
-
Mathematical - (-)(*)(+)(/)(**) - ** is power
-
Concatenate two string using +
-
typeof Operator
-
Assignment Operators = += ++ --
-
Comparison Operators < > <= >=.
Template literals (Template strings)
console.log(`Template Literal ${504-34}`);
A bug in JS
console.log('String with \n\
mulitple \n\
lines');
console.log(`String
multiple
lines with template string`);
function testNum(a) {
let result;
if (a > 0) {
result = "positive";
} else {
result = "NOT positive";
}
return result;
}
console.log(testNum(-5));
// expected output: "NOT positive"
JavaScript type coercion explained
-
5 Falsy values => 0, '', undefined, null, NaN.
-
This values get converted to false when we try to Type cast them in Boolean value.
-
other than those all are truthy values.
- & - and
- | - or
- ~ - not
- && - and
- || - or
const expr = "Papayas";
switch (expr) {
case "Oranges":
console.log("Oranges are $0.59 a pound.");
break;
case "Mangoes":
case "Papayas":
console.log("Mangoes and papayas are $2.79 a pound.");
// expected output: "Mangoes and papayas are $2.79 a pound."
break;
default:
console.log(`Sorry, we are out of ${expr}.`);
}
Conditional (ternary) operator
let greeting = (person) => {
let name = person ? person.name : `stranger`;
return `Howdy, ${name}`;
};
console.log(greeting({ name: `Alice` })); // "Howdy, Alice"
console.log(greeting(null)); // "Howdy, stranger"
"use strict";
// Functions
function logger() {
console.log("My name is Jonas");
}
// calling / running / invoking function
logger();
logger();
logger();
function fruitProcessor(apples, oranges) {
const juice = `Juice with ${apples} apples and ${oranges} oranges.`;
return juice;
}
const appleJuice = fruitProcessor(5, 0);
console.log(appleJuice);
const appleOrangeJuice = fruitProcessor(2, 4);
console.log(appleOrangeJuice);
const num = Number("23");
// Function declaration
function calcAge1(birthYeah) {
return 2037 - birthYeah;
}
const age1 = calcAge1(1991);
// Function expression
const calcAge2 = function (birthYeah) {
return 2037 - birthYeah;
};
const age2 = calcAge2(1991);
console.log(age1, age2);
-
In JavaScript, however, functions are not only syntax but also values, which means they can be assigned to variables, stored in the properties of objects or the elements of arrays, passed as arguments to functions, and so on.
-
A function declaration can be called before initialization. (Hoisting)
// Traditional Function
function (a, b) {
let chuck = 42;
return a + b + chuck;
}
// Arrow Function
(a, b) => {
let chuck = 42;
return a + b + chuck;
}
// Functions Calling Other Functions
function cutFruitPieces(fruit) {
return fruit * 4;
}
function fruitProcessor(apples, oranges) {
const applePieces = cutFruitPieces(apples);
const orangePieces = cutFruitPieces(oranges);
const juice = `Juice with ${applePieces} piece of apple and ${orangePieces} pieces of orange.`;
return juice;
}
console.log(fruitProcessor(2, 3));
// Introduction to Arrays
// Array Declaration
const friends = ["Michael", "Steven", "Peter"];
console.log(friends);
const year = new Array(1991, 1984, 2008, 2020);
console.log(friends[0]);
console.log(friends[2]);
console.log(friends.length);
console.log(friends[friends.length - 1]);
friends[2] = "Jay";
console.log(friends);
// friends = ['Bob', 'Alice']
const firstName = "Jonas";
// Array inside array
const jonas = [firstName, "Schmedtmann", 2037 - 1991, "teacher", friends];
console.log(jonas);
console.log(jonas.length);
// Create an Array
let fruits = ["Apple", "Banana"];
console.log(fruits.length); // 2
// Access an Array item using the index position
let first = fruits[0]; // Apple
let last = fruits[fruits.length - 1]; // Banana
// Loop over an Array
fruits.forEach(function (item, index, array) {
console.log(item, index);
});
// Apple 0
// Banana 1
// Add an item to the end of an Array
let newLength = fruits.push("Orange");
// ["Apple", "Banana", "Orange"]
// Remove an item from the end of an Array
let last = fruits.pop(); // remove Orange (from the end)
// ["Apple", "Banana"]
// Remove an item from the beginning of an Array
let first = fruits.shift(); // remove Apple from the front
// ["Banana"]
// Add an item to the beginning of an Array
let newLength = fruits.unshift("Strawberry"); // add to the front
// ["Strawberry", "Banana"]
// Find the index of an item in the Array
fruits.push("Mango");
// ["Strawberry", "Banana", "Mango"]
let pos = fruits.indexOf("Banana"); // 1
// Remove an item by index position
let removedItem = fruits.splice(pos, 1); // this is how to remove an item
// ["Strawberry", "Mango"]
// Remove items from an index position
let vegetables = ["Cabbage", "Turnip", "Radish", "Carrot"];
console.log(vegetables);
// ["Cabbage", "Turnip", "Radish", "Carrot"]
let pos = 1;
let n = 2;
let removedItems = vegetables.splice(pos, n);
// this is how to remove items, n defines the number of items to be removed,
// starting at the index position specified by pos and progressing toward the end of array.
console.log(vegetables);
// ["Cabbage", "Carrot"] (the original array is changed)
console.log(removedItems);
// ["Turnip", "Radish"]
// Copy an Array
let shallowCopy = fruits.slice(); // this is how to make a copy
// ["Strawberry", "Mango"]
// how to check specific element is present in array
if (friends.includes("Steven")) {
console.log("You have a friend called Steven");
}
- Array is an object too.
const jonas = {
firstName: "Jonas",
lastName: "Schmedtmann",
age: 2037 - 1991,
job: "teacher",
friends: ["Michael", "Peter", "Steven"],
};
// Dot vs. Bracket Notation
const person1 = {};
person1["firstname"] = "Mario";
person1["lastname"] = "Rossi";
console.log(person1.firstname);
// expected output: "Mario"
const person2 = {
firstname: "John",
lastname: "Doe",
};
console.log(person2["lastname"]);
// expected output: "Doe"
const jonas = {
firstName: "Jonas",
lastName: "Schmedtmann",
age: 2037 - 1991,
job: "teacher",
friends: ["Michael", "Peter", "Steven"],
};
console.log(jonas);
console.log(jonas.lastName);
console.log(jonas["lastName"]);
const nameKey = "Name";
console.log(jonas["first" + nameKey]);
console.log(jonas["last" + nameKey]);
// console.log(jonas.'last' + nameKey)
const interestedIn = prompt(
"What do you want to know about Jonas? Choose between firstName, lastName, age, job, and friends"
);
if (jonas[interestedIn]) {
console.log(jonas[interestedIn]);
} else {
console.log(
"Wrong request! Choose between firstName, lastName, age, job, and friends"
);
}
jonas.location = "Portugal";
jonas["twitter"] = "@jonasschmedtman";
console.log(jonas);
// Object Methods
const jonas = {
firstName: "Jonas",
lastName: "Schmedtmann",
birthYeah: 1991,
job: "teacher",
friends: ["Michael", "Peter", "Steven"],
hasDriversLicense: true,
// calcAge: function (birthYeah) {
// return 2037 - birthYeah;
// }
// calcAge: function () {
// console.log(this);
// return 2037 - this.birthYeah;
// }
calcAge: function () {
this.age = 2037 - this.birthYeah;
return this.age;
},
getSummary: function () {
return `${this.firstName} is a ${this.calcAge()}-year old ${
jonas.job
}, and he has ${this.hasDriversLicense ? "a" : "no"} driver's license.`;
},
};
console.log(jonas.calcAge());
console.log(jonas.age);
console.log(jonas.getSummary());
-
this is equal to object calling the method
-
loop in loop
for (let exercise = 4; exercise >= 1; exercise--) {
console.log(`-------- Starting exercise ${exercise}`);
for (let rep = 1; rep < 6; rep++) {
console.log(`Exercise ${exercise}: Lifting weight repetition ${rep} 🏋️♀️`);
}
}
- alt + down arrow key => move your line up
- alt + up arrow key => move your line down
- ctrl + d => select next occurence
- prettier extension
- format on save
- npm live-server
- Google StackOverflow MDN
- Debugging Debugger in chrome
- console.table
- console.warn
- console.error
- debugger keyword
- ctrl + k - clear terminal
- Document Object Model - Structutred representation of HTML documents. Allows javascript to access html elements and styles to manipulate them.
- can change text, HTML attributes, and even CSS styles.
- Tree Structure, generated by browser on HTML load.
- DOM Methods and Properties for DOM Manipulation are not part of JS
- DOM Methods and Properties WEB API's
-
querySelector(.class or #id)
-
querySelectorAll(.class or #id) // nodelist
-
.textContent
-
.src for img
-
.value for input likewise experiment and learn
-
getElementByID(id)
-
document.createElement()
-
Node.appendChild()
-
createTextNode()
-
addEventListener(type,function);
-
type - keypress, keyup, keydown, click, etc.
-
event object can be accessed in event handler function => function (event)
-
.style.cssPropertyName (propertyName should be in camelCase)(Inline Style)
-
.setAttribute()
-
.removeAttribute()
-
.getAttribute()
-
.hasAttribute()
- .classlist.remove() // multiple classes can also be added with , and no need to add . with classname
- .classlist.add()
- .classlist.contains() // contains class or not
- .classlist.toggle() // toggle
-
Javascript is a High-Level, Prototype-Based, Object-Oriented, Multi-Paradigm, Interpreted or Just in time compiled, Dynamic, Single Threaded, Garbage Collected programming language with First-class functin and a non-blocking event loop concurrency model.
-
Garbage Collected - Memory Management
-
Interpreted or just-in-time compiled = Just-in-time compilation
-
Paradigm = An approach and mindset of structuring code, which will direct your coding style and technique. (Imperative vs. Declarative).
- Procedural Programming
- Object Oriented Programming (OOP)
- Functional Programming (FP)
-
Prototype-based object-oriented = Prototype-based programming is a style of object-oriented programming in which classes are not explicitly defined, but rather derived by adding properties and methods to an instance of another class or, less frequently, adding them to an empty object.
-
First-class Functions = In a language with first-class functions, functions are simply treated as variables. We can pass them into other functions, and return them from functions.
-
Dynamic = JavaScript is a loosely typed and dynamic language. Variables in JavaScript are not directly associated with any particular value type, and any variable can be assigned (and re-assigned) values of all types:
let foo = 42; // foo is now a number
foo = "bar"; // foo is now a string
foo = true; // foo is now a boolean
- JavaScript Engine => The JavaScript engine is an interpreter that parses and executes a JavaScript program.(Google's - V8 Engine)
- Compilation Vs. Interpretation
- Compilation: Entire code is converted into machine code at once, and written to a binary file that can be excecuted by a computer.
- Interpretation: Interpreter runs through the source code and executes it line by line.
- Just-in-time(JIT) compilation: Entire code is converted into machine code at once, then executed immediately.
- Parsing and AST
- compilation Execution and Optimization
- Javascript Runtime
- Web API
- Heap, CallBack, Callback Queue, Event Loop
- Creation of Global Execution context(for top level code)
- Execution Context: Environment in which a piece of JavaScript is executed. Stores all necessary information for some code to be executed.
- Execution of top-level-code(inside global EC)
- Execution of functions and waiting for callbacks
- Exactly one global execution context (EC): Default context, created for code that is not inside any function(top-level).
- One execution context per function: For each function call, a new execution context is created. All together make the call stack
- generated during "creation phase", right before execution
- Variable Environment
- let, const and var declarations
- functions
- arguments object (Not in arrow function)
- Scope chain
- this keyword - (Not in arrow function)
- Scoping: How our program's variables are organized and accessed.
- Lexical Scoping: Scoping is controlled by placement of functions and blocks in the code.
- Scope: Space or environment in which a certain variable is declared (variable environment in case of functions). There is global scope, function scope, and block scope.
- Scope of a variable: Region of our code where a certain varible can be accessed.
- Global Scope: Variable declared in global scope are accessible everywhere.
- Function Scope: Variables are accessible only inside function, not outside. Also called local scope. var is function scoped.
- Block Scope: Variables are accessible only inside block(block scoped). However, this only applies to let and const variables. Functions are also block scoped(only in strict mode).
Understanding scope and scope chain
- Makes some types of variable accessible/usable in the code before they are actually declared. "Variables lifted to the top of their scope".
- Before execution, Code is scanned for variable declarations and for each variable a new property is created in the variable environment object.
-
Hoisted? Initial Value Scope function declarations [x]Yes Actual function Block(in strict mode) var variables [x]Yes undefined Function let and const variables [ ]No uninitialized, Temporarily dead zone Block function expressions and arrows Depends if using var or let/ const
const myName = "Jonas";
if (myName === "Jonas") {
console.log(`Jonas is a ${job}`); --- we get reference error
const age = 2037 - 1989; --- This three lines are TDZ for job variable
console.log(age); ---
const job = "teacher";
console.log(x);
}
-
why TDZ?
- Makes it easier to avoid and catch errors: accessing variables vefore declaration is bad practice and should be avoided.
- Makes const variables actually work.
-
Why Hoisting?
- Using functions before actual declaration;
- var hoisting is just a byproduct.
// Functions
console.log(addDecl(2, 3)); // 5
// console.log(addExpr(2, 3)); // cannot access before initialization
console.log(addArrow);
// console.log(addArrow(2, 3)); // addArrow is not a function since addArrow is undefined. since it is var.
function addDecl(a, b) {
return a + b;
}
const addExpr = function (a, b) {
return a + b;
};
var addArrow = (a, b) => a + b;
var x = 1;
let y = 2;
const z = 3;
console.log(x === window.x); // true
console.log(y === window.y); // false
console.log(z === window.z); // false
-
this keyword - Special variable that is created for everu execution context (every function). Takes the value of (points to) the "owner" of the function in which the this keyword is used.
-
this is NOT static. It depends on how the function is called, and its value is only assigned when the function is actually called.
-
this Method this = Object that is calling the method Simple function call this = undefined (In strict mode! otherwise: window(in the browser)) Arrow functions this = this of sunrrouding function (lexical this). AF don't get own this keyword. Event listener this = DOM element that the handler is attached to new,call,apply,bind
const jonas = {
year: 1991,
calcAge: function () {
console.log(this);
console.log(2037 - this.year);
},
};
jonas.calcAge(); // 46
const matilda = {
year: 2017,
};
matilda.calcAge = jonas.calcAge;
matilda.calcAge(); // 20
const f = jonas.calcAge;
f(); // undefined
// var firstName = "Matilda"; if we uncomment it greet will print firstName
const jonas = {
firstName: "Jonas",
year: 1991,
calcAge: function () {
console.log(this); // jonas object
console.log(2037 - this.year); // 46
},
greet: () => {
console.log(this); // window
console.log(`Hey ${this.firstName}`);
},
};
jonas.greet();
const jonas = {
firstName: "Jonas",
year: 1991,
calcAge: function () {
console.log(this); // jonas object
console.log(2037 - this.year);
// Problem
const isMillenial = function () {
console.log(this); // undefined
console.log(this.year >= 1981 && this.year <= 1996);
};
isMillenial(); // calls function
// Solution 1
const self = this; // self or that
const isMillenial = function () {
console.log(self); // jonas object
console.log(self.year >= 1981 && self.year <= 1996);
};
isMillenial(); // calls function
// Solution 2
const isMillenial = () => {
console.log(this); // jonas object
console.log(this.year >= 1981 && this.year <= 1996);
};
isMillenial(); // calls function
},
greet: () => {
console.log(this);
console.log(`Hey ${this.firstName}`);
},
};
jonas.greet();
jonas.calcAge();
// arguments keyword
const addExpr = function (a, b) {
console.log(arguments); // parameters
return a + b;
};
addExpr(2, 5);
addExpr(2, 5, 8, 12);
var addArrow = (a, b) => {
console.log(arguments); // arguments not for arrow functions
return a + b;
};
addArrow(2, 5, 8);
-
// Objects vs. primitives let age = 30; /// stored in stack let oldAge = age; // points to age address age = 31; // age points to new value with diff address console.log(age); // 31 console.log(oldAge); // 30 const me = { // stored in heap // me points to reference address and stored in stack name: "Jonas", age: 30, }; const friend = me; // points to me address friend.age = 27; console.log("Friend:", friend); // 27 console.log("Me", me); // 27
const arr = [2, 3, 4];
const [x, y, z] = arr;
console.log(x, y, z);
// we can also take first and second only
const [x, y] = [2, 3, 4];
console.log(x, y); // 2, 3
// if we want to skip one just make blank space for that
const [x, , z] = [2, 3, 4];
console.log(x, z); // 2, 4
// swap elements
[main, secondary] = [secondary, main];
// nested destructuring
const nested = [2, 4, [5, 6]];
const [i, , j] = nested;
console.log(i, j); // 2,[5,6]
const [i, , [j, k]] = nested;
console.log(i, j, k); // 2,5,6
// default values
const [p, q, r] = [8, 9];
console.log(p, q, r); // 8 9 undefined
// default values
const [p = 1, q = 1, r = 1] = [8, 9];
console.log(p, q, r); // 8 9 1
const restaurant = {
name: "Classico Italiano",
location: "Via Angelo Tavanti 23, Firenze, Italy",
categories: ["Italian", "Pizzeria", "Vegetarian", "Organic"],
starterMenu: ["Focaccia", "Bruschetta", "Garlic Bread", "Caprese Salad"],
mainMenu: ["Pizza", "Pasta", "Risotto"],
openingHours: {
thu: {
open: 12,
close: 22,
},
fri: {
open: 11,
close: 23,
},
sat: {
open: 0,
close: 24,
},
},
order: function (starterIndex, mainIndex) {
return [this.starterMenu[starterIndex], this.mainMenu[mainIndex]];
},
// passing object
orderDelivery: function (obj) {
console.log(obj);
},
// for destructuring object
orderDelivery: function ({ starterIndex, mainIndex, time, address }) {
// destructuring right here
console.log(
`order received! ${this.starterMenu[starterIndex]} and ${this.mainMenu[mainIndex]} will be delivered to ${address} at ${time}`
);
},
// for having default values
orderDelivery: function ({
starterIndex = 1,
mainIndex = 0,
time = "20:00",
address,
}) {
console.log(
`Order received! ${this.starterMenu[starterIndex]} and ${this.mainMenu[mainIndex]} will be delivered to ${address} at ${time}`
);
},
orderPasta: function (ing1, ing2, ing3) {
console.log(
`Here is your delicious pasta with ${ing1}, ${ing2} and ${ing3}`
);
},
orderPizza: function (mainIngredient, ...otherIngredients) {
console.log(mainIngredient);
console.log(otherIngredients);
},
};
// Destructuring Objects
restaurant.orderDelivery({
time: "22:30",
address: "Via del Sole, 21",
mainIndex: 2,
starterIndex: 2,
});
// Default values
restaurant.orderDelivery({
address: "Via del Sole, 21",
starterIndex: 1,
});
// Object Destructuring
const { name, openingHours, categories } = restaurant;
console.log(name, openingHours, categories);
// changing names
const {
name: restaurantName,
openingHours: hours,
categories: tags,
} = restaurant;
console.log(restaurantName, hours, tags);
// Default values
const { menu = [], starterMenu: starters = [] } = restaurant;
console.log(menu, starters);
// Mutating variables
let a = 111;
let b = 999;
const obj = { a: 23, b: 7, c: 14 };
// {a,b}=obj; // unexpected token =
({ a, b } = obj);
console.log(a, b);
// Nested objects
const { fri } = openingHours;
const {
fri: { open, close },
} = openingHours;
console.log(open, close);
const {
fri: { open: o, close: c },
} = openingHours;
console.log(o, c);
// The Spread Operator (...)
const arr = [7, 8, 9];
const badNewArr = [1, 2, arr[0], arr[1], arr[2]];
console.log(badNewArr);
const newArr = [1, 2, ...arr];
console.log(newArr);
console.log(...newArr);
console.log(1, 2, 7, 8, 9);
const newMenu = [...restaurant.mainMenu, "Gnocci"];
console.log(newMenu);
// Copy array shallow
const mainMenuCopy = [...restaurant.mainMenu];
// Join 2 arrays
const menu = [...restaurant.starterMenu, ...restaurant.mainMenu];
console.log(menu);
// Iterables: arrays, strings, maps, sets. NOT objects
const str = "Jonas";
const letters = [...str, " ", "S."];
console.log(letters);
console.log(...str);
// console.log(`${...str} Schmedtmann`); // unexpected token
// Real-world example
const ingredients = [
// prompt("Let's make pasta! Ingredient 1?"),
// prompt('Ingredient 2?'),
// prompt('Ingredient 3'),
];
console.log(ingredients);
restaurant.orderPasta(ingredients[0], ingredients[1], ingredients[2]);
restaurant.orderPasta(...ingredients);
// Objects
const newRestaurant = { foundedIn: 1998, ...restaurant, founder: "Guiseppe" };
console.log(newRestaurant);
const restaurantCopy = { ...restaurant };
restaurantCopy.name = "Ristorante Roma";
console.log(restaurantCopy.name);
console.log(restaurant.name);
// Rest Pattern
// SPREAD, because on RIGHT side of =
const arr = [1, 2, ...[3, 4]];
// REST, because on LEFT side of =
const [a, b, ...others] = [1, 2, 3, 4, 5];
console.log(a, b, others);
const [pizza, , risotto, ...otherFood] = [
...restaurant.mainMenu,
...restaurant.starterMenu,
];
console.log(pizza, risotto, otherFood);
// Objects
const { sat, ...weekdays } = restaurant.openingHours;
console.log(weekdays);
// 2) Functions - Rest Parameters
const add = function (...numbers) {
let sum = 0;
for (let i = 0; i < numbers.length; i++) sum += numbers[i];
console.log(sum);
};
add(2, 3);
add(5, 3, 7, 2);
add(8, 2, 5, 3, 2, 1, 4);
const x = [23, 5, 7];
add(...x);
restaurant.orderPizza("mushrooms", "onion", "olives", "spinach");
restaurant.orderPizza("mushrooms");
// Short Circuiting (&& and ||)
console.log("---- OR ----");
// Use ANY data type, return ANY data type, short-circuiting
console.log(3 || "Jonas"); // 3
console.log("" || "Jonas"); // Jonas
console.log(true || 0); // true
console.log(undefined || null); // null
console.log(undefined || 0 || "" || "Hello" || 23 || null); // Hello
restaurant.numGuests = 0;
const guests1 = restaurant.numGuests ? restaurant.numGuests : 10;
console.log(guests1);
const guests2 = restaurant.numGuests || 10;
console.log(guests2);
console.log("---- AND ----");
console.log(0 && "Jonas"); // 0
console.log(7 && "Jonas"); // Jonas
console.log("Hello" && 23 && null && "jonas"); // null
// Practical example
if (restaurant.orderPizza) {
restaurant.orderPizza("mushrooms", "spinach");
}
restaurant.orderPizza && restaurant.orderPizza("mushrooms", "spinach");
// The Nullish Coalescing Operator
restaurant.numGuests = 0;
const guests = restaurant.numGuests || 10;
console.log(guests);
// Nullish: null and undefined (NOT 0 or '')
const guestCorrect = restaurant.numGuests ?? 10; // 0
console.log(guestCorrect);
- for Of Loop
const menu = [1, 2, 4, 26, 52, 5, 5, 342, 342, 234, 63];
for (const item of menu) console.log(item);
console.log(...menu.entries()); // Array Iterators
const openingHours = {
th: {
open: 2,
close: 22,
},
sat: {
open: 1,
close: 22,
},
};
const restaurant = {
name: "Dhaba",
// ES 5 way
openingHours: openingHours,
// ES 6 way
openingHours,
// function ES 5
order: function (starterIndex, mainIndex) {
return [this.starterMenu[starterIndex], this.mainMenu[mainIndex]];
},
// function ES 6
ordesr(starterIndex, mainIndex) {
return [this.starterMenu[starterIndex], this.mainMenu[mainIndex]];
},
};
// use template literal
const weekdays = ["mon", "tue", "wed", "thu", "fri"];
const op = {
[weedays[4]]: {
open: 13,
close: 23,
},
[`day-${3 + 21}`]: {
open: 2,
close: 22,
},
};
if (restaurant.openingHours && restaurant.openingHours.mon)
console.log(restaurant.openingHours.mon.open);
// we get error
console.log(restaurant.openingHours.mon.open);
// With Optional Chaining
cosole.log(restaurant.openingHours.mon?.open); //undefined
cosole.log(restaurant.openingHours?.mon?.open);
// Example
const days = ["mon", "tue", "wed", "thu", "fri", "sat", "sun"];
for (const day of days) {
const open = restaurant.openingHours[day]?.open ?? "closed";
console.log(`On ${day}, we open at ${open}`);
}
// Methods
console.log(restaurant.order?.(0, 1) ?? "Method does not exist");
console.log(restaurant.orderRisotto?.(0, 1) ?? "Method does not exist");
// Arrays
const users = [{ name: "Jonas", email: "hello@jonas.io" }];
// const users = [];
console.log(users[0]?.name ?? "User array empty");
if (users.length > 0) console.log(users[0].name);
else console.log("user array empty");
// Property NAMES
const properties = Object.keys(openingHours);
console.log(properties);
let openStr = `We are open on ${properties.length} days: `;
for (const day of properties) {
openStr += `${day}, `;
}
console.log(openStr);
// Property VALUES
const values = Object.values(openingHours);
console.log(values);
// Entire object
const entries = Object.entries(openingHours);
// console.log(entries);
// [key, value]
for (const [day, { open, close }] of entries) {
console.log(`On ${day} we open at ${open} and close at ${close}`);
}
const ordersSet = new Set([
"Pasta",
"Pizza",
"Pizza",
"Risotto",
"Pasta",
"Pizza",
]);
console.log(ordersSet);
console.log(new Set("Jonas"));
console.log(ordersSet.size);
console.log(ordersSet.has("Pizza"));
console.log(ordersSet.has("Bread"));
ordersSet.add("Garlic Bread");
ordersSet.add("Garlic Bread");
ordersSet.delete("Risotto");
// ordersSet.clear();
console.log(ordersSet);
for (const order of ordersSet) console.log(order);
// Example
const staff = ["Waiter", "Chef", "Waiter", "Manager", "Chef", "Waiter"];
const staffUnique = [...new Set(staff)];
console.log(staffUnique);
console.log(
new Set(["Waiter", "Chef", "Waiter", "Manager", "Chef", "Waiter"]).size
);
console.log(new Set("jonasschmedtmann").size);
// Maps: Fundamentals
const rest = new Map();
rest.set("name", "Classico Italiano");
rest.set(1, "Firenze, Italy");
console.log(rest.set(2, "Lisbon, Portugal"));
rest
.set("categories", ["Italian", "Pizzeria", "Vegetarian", "Organic"])
.set("open", 11)
.set("close", 23)
.set(true, "We are open :D")
.set(false, "We are closed :(");
console.log(rest.get("name"));
console.log(rest.get(true));
console.log(rest.get(1));
const time = 8;
console.log(rest.get(time > rest.get("open") && time < rest.get("close")));
console.log(rest.has("categories"));
rest.delete(2);
// rest.clear();
const arr = [1, 2];
rest.set(arr, "Test");
rest.set(document.querySelector("h1"), "Heading");
console.log(rest);
console.log(rest.size);
console.log(rest.get(arr));
// Maps: Iteration
const question = new Map([
["question", "What is the best programming language in the world?"],
[1, "C"],
[2, "Java"],
[3, "JavaScript"],
["correct", 3],
[true, "Correct 🎉"],
[false, "Try again!"],
]);
console.log(question);
// Convert object to map
console.log(Object.entries(openingHours));
const hoursMap = new Map(Object.entries(openingHours));
console.log(hoursMap);
// Quiz app
console.log(question.get("question"));
for (const [key, value] of question) {
if (typeof key === "number") console.log(`Answer ${key}: ${value}`);
}
// const answer = Number(prompt('Your answer'));
const answer = 3;
console.log(answer);
console.log(question.get(question.get("correct") === answer));
// Convert map to array
console.log([...question]);
// console.log(question.entries());
console.log([...question.keys()]);
console.log([...question.values()]);
Arrays | Sets |
---|---|
Use when you need ordered list of values (might contain duplicates) | Use when you need to work with unique values |
Use when you need to manipulate data | Use when high-performance is really important |
Use to remove duplicates from arrays |
Objects | Maps |
---|---|
More "traditional" key/value store("abused" objects) | Better performance |
Easier to write and access values with . and [] | Keys can have any data type |
Easy to iterate | |
Easy to compute size | |
Use when you need to include functions(methods) | Use when you simply need to map key to values |
Use when working with JSON (can covert to map) | Use when you need keys that are not strings |
// Working With Strings - Part 1
const airline = "TAP Air Portugal";
const plane = "A320";
console.log(plane[0]);
console.log(plane[2]);
console.log("B737"[0]);
console.log(airline.length);
console.log("B737".length);
console.log(airline.indexOf("r"));
console.log(airline.lastIndexOf("r"));
console.log(airline.indexOf("portugal"));
console.log(airline.slice(4));
console.log(airline.slice(4, 7));
console.log(airline.slice(0, airline.indexOf(" ")));
console.log(airline.slice(airline.lastIndexOf(" ") + 1));
console.log(airline.slice(-2));
console.log(airline.slice(1, -1));
const checkMiddleSeat = function (seat) {
// B and E are middle seats
const s = seat.slice(-1);
if (s === "B" || s === "E") console.log("You got the middle seat 😬");
else console.log("You got lucky 😎");
};
checkMiddleSeat("11B");
checkMiddleSeat("23C");
checkMiddleSeat("3E");
console.log(new String("jonas"));
console.log(typeof new String("jonas"));
console.log(typeof new String("jonas").slice(1));
- methods aare called on objects like arrays, strings beu\ing premitives js coverts behind the scene to string object.(boxing)
- after the operation it coverts back to string.
// Working With Strings - Part 2
const airline = "TAP Air Portugal";
console.log(airline.toLowerCase());
console.log(airline.toUpperCase());
// Fix capitalization in name
const passenger = "jOnAS"; // Jonas
const passengerLower = passenger.toLowerCase();
const passengerCorrect =
passengerLower[0].toUpperCase() + passengerLower.slice(1);
console.log(passengerCorrect);
// Comparing emails
const email = "hello@jonas.io";
const loginEmail = " Hello@Jonas.Io \n";
// const lowerEmail = loginEmail.toLowerCase();
// const trimmedEmail = lowerEmail.trim();
const normalizedEmail = loginEmail.toLowerCase().trim();
console.log(normalizedEmail);
console.log(email === normalizedEmail);
// replacing
const priceGB = "288,97£";
const priceUS = priceGB.replace("£", "$").replace(",", ".");
console.log(priceUS);
const announcement =
"All passengers come to boarding door 23. Boarding door 23!";
console.log(announcement.replace("door", "gate"));
// console.log(announcement.replaceAll('door', 'gate'));
console.log(announcement.replace(/door/g, "gate"));
// regExp
// Booleans
const plane = "Airbus A320neo";
console.log(plane.includes("A320"));
console.log(plane.includes("Boeing"));
console.log(plane.startsWith("Airb"));
if (plane.startsWith("Airbus") && plane.endsWith("neo")) {
console.log("Part of the NEW ARirbus family");
}
// Working With Strings - Part 3
// Split and join
console.log("a+very+nice+string".split("+"));
console.log("Jonas Schmedtmann".split(" "));
const [firstName, lastName] = "Jonas Schmedtmann".split(" ");
const newName = ["Mr.", firstName, lastName.toUpperCase()].join(" ");
console.log(newName);
const capitalizeName = function (name) {
const names = name.split(" ");
const namesUpper = [];
for (const n of names) {
// namesUpper.push(n[0].toUpperCase() + n.slice(1));
namesUpper.push(n.replace(n[0], n[0].toUpperCase()));
}
console.log(namesUpper.join(" "));
};
capitalizeName("jessica ann smith davis");
capitalizeName("jonas schmedtmann");
// Padding
const message = "Go to gate 23!";
console.log(message.padStart(20, "+").padEnd(30, "+"));
console.log("Jonas".padStart(20, "+").padEnd(30, "+"));
const maskCreditCard = function (number) {
const str = number + "";
const last = str.slice(-4);
return last.padStart(str.length, "*");
};
console.log(maskCreditCard(64637836));
console.log(maskCreditCard(43378463864647384));
console.log(maskCreditCard("334859493847755774747"));
// Repeat
const message2 = "Bad waether... All Departues Delayed... ";
console.log(message2.repeat(5));
const planesInLine = function (n) {
console.log(`There are ${n} planes in line ${"🛩".repeat(n)}`);
};
planesInLine(5);
planesInLine(3);
planesInLine(12);
// Default Parameters
const bookings = [];
const createBooking = function (
flightNum,
numPassengers = 1,
price = 199 * numPassengers
) {
// ES5
// numPassengers = numPassengers || 1;
// price = price || 199;
const booking = {
flightNum,
numPassengers,
price,
};
console.log(booking);
bookings.push(booking);
};
createBooking("LH123");
createBooking("LH123", 2, 800);
createBooking("LH123", 2);
createBooking("LH123", 5);
createBooking("LH123", undefined, 1000);
// How Passing Arguments Works: Values vs. Reference
const flight = "LH234";
const jonas = {
name: "Jonas Schmedtmann",
passport: 24739479284,
};
const checkIn = function (flightNum, passenger) {
flightNum = "LH999";
passenger.name = "Mr. " + passenger.name;
if (passenger.passport === 24739479284) {
alert("Checked in");
} else {
alert("Wrong passport!");
}
};
// checkIn(flight, jonas);
// console.log(flight); // not changed
// console.log(jonas); // changed
// Is the same as doing...
// const flightNum = flight;
// const passenger = jonas;
const newPassport = function (person) {
person.passport = Math.trunc(Math.random() * 100000000000);
};
newPassport(jonas);
checkIn(flight, jonas);
- JavaScript does not have pass by reference.
First Class Functions | Higher Order Functions |
---|---|
JavaScript treats functions as first-class citizens | A function that receives another function as an argument that returns a new function or both |
This means that functions are simply values | This is only possible because of first class functions |
Functions are just another 'type' of object | 1) Fuction that receives another function. Ex- addEventListner is higher order function since it receives a callback function |
store functions in variables or properties | 2) function that returns a new function |
pass function as arguments to other functions | |
return functions from functions | |
call methods on functions |
// Functions Accepting Callback Functions
const oneWord = function (str) {
return str.replace(/ /g, "").toLowerCase();
};
const upperFirstWord = function (str) {
const [first, ...others] = str.split(" ");
return [first.toUpperCase(), ...others].join(" ");
};
// Higher-order function
const transformer = function (str, fn) {
console.log(`Original string: ${str}`);
console.log(`Transformed string: ${fn(str)}`);
console.log(`Transformed by: ${fn.name}`); // function property
};
transformer("JavaScript is the best!", upperFirstWord);
transformer("JavaScript is the best!", oneWord);
// JS uses callbacks all the time
const high5 = function () {
console.log("👋");
};
document.body.addEventListener("click", high5);
["Jonas", "Martha", "Adam"].forEach(high5);
- Javascript allows us to create abstraction.
// Functions Returning Functions
const greet = function (greeting) {
return function (name) {
console.log(`${greeting} ${name}`);
};
};
const greeterHey = greet("Hey");
greeterHey("Jonas");
greeterHey("Steven");
greet("Hello")("Jonas");
// Challenge
const greetArr = (greeting) => (name) => console.log(`${greeting} ${name}`);
greetArr("Hi")("Jonas");
- Important in functional programming paradigm.
// The call and apply Methods
const lufthansa = {
airline: "Lufthansa",
iataCode: "LH",
bookings: [],
// book: function() {}
book(flightNum, name) {
console.log(
`${name} booked a seat on ${this.airline} flight ${this.iataCode}${flightNum}`
);
this.bookings.push({ flight: `${this.iataCode}${flightNum}`, name });
},
};
lufthansa.book(239, "Jonas Schmedtmann");
lufthansa.book(635, "John Smith");
const eurowings = {
airline: "Eurowings",
iataCode: "EW",
bookings: [],
};
const book = lufthansa.book;
// Does NOT work
// book(23, 'Sarah Williams');
// Call method
book.call(eurowings, 23, "Sarah Williams");
console.log(eurowings);
book.call(lufthansa, 239, "Mary Cooper");
console.log(lufthansa);
const swiss = {
airline: "Swiss Air Lines",
iataCode: "LX",
bookings: [],
};
book.call(swiss, 583, "Mary Cooper");
// Apply method
const flightData = [583, "George Cooper"];
book.apply(swiss, flightData);
console.log(swiss);
// instead of apply you can do like this
book.call(swiss, ...flightData);
// The bind Method
// book.call(eurowings, 23, 'Sarah Williams');
const bookEW = book.bind(eurowings);
const bookLH = book.bind(lufthansa);
const bookLX = book.bind(swiss);
bookEW(23, "Steven Williams");
const bookEW23 = book.bind(eurowings, 23);
bookEW23("Jonas Schmedtmann");
bookEW23("Martha Cooper");
// With Event Listeners
lufthansa.planes = 300;
lufthansa.buyPlane = function () {
console.log(this);
this.planes++;
console.log(this.planes);
};
// lufthansa.buyPlane();
document
.querySelector(".buy")
.addEventListener("click", lufthansa.buyPlane.bind(lufthansa));
// Partial application
const addTax = (rate, value) => value + value * rate;
console.log(addTax(0.1, 200));
const addVAT = addTax.bind(null, 0.23);
// addVAT = value => value + value * 0.23;
console.log(addVAT(100));
console.log(addVAT(23));
// use function return function and recreate above
const addTaxRate = function (rate) {
return function (value) {
return value + value * rate;
};
};
const addVAT2 = addTaxRate(0.23);
console.log(addVAT2(100));
console.log(addVAT2(23));
// Immediately Invoked Function Expressions (IIFE)
const runOnce = function () {
console.log("This will never run again");
};
runOnce();
// IIFE
(function () {
console.log("This will never run again");
const isPrivate = 23;
})();
// console.log(isPrivate);
(() => console.log("This will ALSO never run again"))();
{
const isPrivate = 23;
var notPrivate = 46;
}
// console.log(isPrivate);
console.log(notPrivate);
- A function has access to the variable environment (VE) of the execution context in which it was created.
- Closure: VE attached to the function, exactly as it was at the time and place the function was created.
- A closure is the closed-over variable environment of the execution context in which a function was created, even after that execution is gone.
- (Less Formal): A closure gives a function access to all the variables of its parent function, even after that parent function has returned. The function keeps a reference to its outer scope, which preserves the scope chain throughout time.
- (Less Formal): A closure makes sure that a function doesn't loose connection to variables that existed at the function's birth place.
- (Less formal): A closure is like a backpack that a function carries around wherever it goes. This backpack has all the variables that were present in the environment where the function was created.
- we do not have to manually create closures, this is a javascript feature that happens automatically, we can't even access closed-over variables explicitly. A closure is not a tangible JavaScript object.
// Closures
const secureBooking = function () {
let passengerCount = 0;
return function () {
passengerCount++;
console.log(`${passengerCount} passengers`);
};
};
const booker = secureBooking();
booker();
booker();
booker();
console.dir(booker);
// More Closure Examples
// Example 1
let f;
const g = function () {
const a = 23;
f = function () {
console.log(a * 2);
};
};
const h = function () {
const b = 777;
f = function () {
console.log(b * 2);
};
};
g();
f();
console.dir(f);
// Re-assigning f function
h();
f();
console.dir(f);
// Example 2
const boardPassengers = function (n, wait) {
const perGroup = n / 3;
setTimeout(function () {
console.log(`We are now boarding all ${n} passengers`);
console.log(`There are 3 groups, each with ${perGroup} passengers`);
}, wait * 1000);
console.log(`Will start boarding in ${wait} seconds`);
};
const perGroup = 1000; // closure have higer priority over scope chain
boardPassengers(180, 3);
- Arrays are Also Objects
// Simple Array Methods
let arr = ["a", "b", "c", "d", "e"];
// SLICE
console.log(arr.slice(2));
console.log(arr.slice(2, 4));
console.log(arr.slice(-2));
console.log(arr.slice(-1));
console.log(arr.slice(1, -2));
console.log(arr.slice());
console.log([...arr]);
// SPLICE mutates original array
// console.log(arr.splice(2));
arr.splice(-1);
console.log(arr);
arr.splice(1, 2);
console.log(arr);
// REVERSE mutates original array
arr = ["a", "b", "c", "d", "e"];
const arr2 = ["j", "i", "h", "g", "f"];
console.log(arr2.reverse());
console.log(arr2);
// CONCAT
const letters = arr.concat(arr2);
console.log(letters);
console.log([...arr, ...arr2]);
// JOIN
console.log(letters.join(" - "));
// Looping Arrays: forEach
const movements = [200, 450, -400, 3000, -650, -130, 70, 1300];
// for (const movement of movements) {
for (const [i, movement] of movements.entries()) {
if (movement > 0) {
console.log(`Movement ${i + 1}: You deposited ${movement}`);
} else {
console.log(`Movement ${i + 1}: You withdrew ${Math.abs(movement)}`);
}
}
console.log("---- FOREACH ----");
movements.forEach(function (mov, i, arr) {
if (mov > 0) {
console.log(`Movement ${i + 1}: You deposited ${mov}`);
} else {
console.log(`Movement ${i + 1}: You withdrew ${Math.abs(mov)}`);
}
});
// 0: function(200)
// 1: function(450)
// 2: function(400)
// ...
// forEach With Maps and Sets
// Map
const currencies = new Map([
["USD", "United States dollar"],
["EUR", "Euro"],
["GBP", "Pound sterling"],
]);
currencies.forEach(function (value, key, map) {
console.log(`${key}: ${value}`);
});
// Set
const currenciesUnique = new Set(["USD", "GBP", "USD", "EUR", "EUR"]);
console.log(currenciesUnique);
currenciesUnique.forEach(function (value, _, map) {
console.log(`${value}: ${value}`);
}); // _ unnecessary parameter
- InsertAdjacentHTML()
- innerHTML
- split()
- map return a new array containing the results of applying an operation on all original array elements
const movements = [200, 450, -400, 3000, -650, -130, 70, 1300];
// The map Method
const eurToUsd = 1.1;
// const movementsUSD = movements.map(function (mov) {
// return mov * eurToUsd;
// });
const movementsUSD = movements.map((mov) => mov * eurToUsd);
console.log(movements);
console.log(movementsUSD);
const movementsUSDfor = [];
for (const mov of movements) movementsUSDfor.push(mov * eurToUsd);
console.log(movementsUSDfor);
const movementsDescriptions = movements.map(
(mov, i) =>
`Movement ${i + 1}: You ${mov > 0 ? "deposited" : "withdrew"} ${Math.abs(
mov
)}`
);
console.log(movementsDescriptions);
- filter returns a new array containing the array elements that passed a specific test condition.
// The filter Method
const deposits = movements.filter(function (mov, i, arr) {
return mov > 0;
});
console.log(movements);
console.log(deposits);
const depositsFor = [];
for (const mov of movements) if (mov > 0) depositsFor.push(mov);
console.log(depositsFor);
const withdrawals = movements.filter((mov) => mov < 0);
console.log(withdrawals);
- reduce boils ('reduces') all array elements down to one single value(eg. adding all elements together).
// The reduce Method
console.log(movements);
// accumulator -> SNOWBALL
// const balance = movements.reduce(function (acc, cur, i, arr) {
// console.log(`Iteration ${i}: ${acc}`);
// return acc + cur;
// }, 0);
const balance = movements.reduce((acc, cur) => acc + cur, 0);
console.log(balance);
let balance2 = 0;
for (const mov of movements) balance2 += mov;
console.log(balance2);
// Maximum value
const max = movements.reduce((acc, mov) => {
if (acc > mov) return acc;
else return mov;
}, movements[0]);
console.log(max);
// The Magic of Chaining Methods
const eurToUsd = 1.1;
console.log(movements);
// PIPELINE
const totalDepositsUSD = movements
.filter((mov) => mov > 0)
.map((mov, i, arr) => {
// console.log(arr);
return mov * eurToUsd;
})
// .map(mov => mov * eurToUsd)
.reduce((acc, mov) => acc + mov, 0);
console.log(totalDepositsUSD);
// The find Method
const firstWithdrawal = movements.find((mov) => mov < 0);
console.log(movements);
console.log(firstWithdrawal);
console.log(accounts);
const account = accounts.find((acc) => acc.owner === "Jessica Davis");
console.log(account);
// some and every
console.log(movements);
// EQUALITY
console.log(movements.includes(-130));
// SOME: CONDITION
console.log(movements.some((mov) => mov === -130));
const anyDeposits = movements.some((mov) => mov > 0);
console.log(anyDeposits);
// EVERY
console.log(movements.every((mov) => mov > 0));
console.log(account4.movements.every((mov) => mov > 0));
// Separate callback
const deposit = (mov) => mov > 0;
console.log(movements.some(deposit));
console.log(movements.every(deposit));
console.log(movements.filter(deposit));
///////////////////////////////////////
// flat and flatMap
const arr = [[1, 2, 3], [4, 5, 6], 7, 8];
console.log(arr.flat());
const arrDeep = [[[1, 2], 3], [4, [5, 6]], 7, 8];
console.log(arrDeep.flat(2));
// flat
const overalBalance = accounts
.map((acc) => acc.movements)
.flat()
.reduce((acc, mov) => acc + mov, 0);
console.log(overalBalance);
// flatMap
const overalBalance2 = accounts
.flatMap((acc) => acc.movements)
.reduce((acc, mov) => acc + mov, 0);
console.log(overalBalance2);
///////////////////////////////////////
// Sorting Arrays
// Strings
const owners = ["Jonas", "Zach", "Adam", "Martha"];
console.log(owners.sort());
console.log(owners);
// Numbers
console.log(movements);
// return < 0, A, B (keep order)
// return > 0, B, A (switch order)
// Ascending
// movements.sort((a, b) => {
// if (a > b) return 1;
// if (a < b) return -1;
// });
movements.sort((a, b) => a - b);
console.log(movements);
// Descending
// movements.sort((a, b) => {
// if (a > b) return -1;
// if (a < b) return 1;
// });
movements.sort((a, b) => b - a);
console.log(movements);
///////////////////////////////////////
// More Ways of Creating and Filling Arrays
const arr = [1, 2, 3, 4, 5, 6, 7];
console.log(new Array(1, 2, 3, 4, 5, 6, 7));
// Empty arrays + fill method
const x = new Array(7); // empty*7
console.log(x);
// console.log(x.map(() => 5));
x.fill(1, 3, 5); // fill with 1 from 3 to 5 index
x.fill(1);
console.log(x);
arr.fill(23, 2, 6);
console.log(arr);
// Array.from
const y = Array.from({ length: 7 }, () => 1);
console.log(y);
// _ is used to indicate does not used in program
const z = Array.from({ length: 7 }, (_, i) => i + 1);
console.log(z);
labelBalance.addEventListener("click", function () {
const movementsUI = Array.from(
document.querySelectorAll(".movements__value"),
(el) => Number(el.textContent.replace("€", ""))
);
console.log(movementsUI);
const movementsUI2 = [...document.querySelectorAll(".movements__value")];
});
///////////////////////////////////////
// Converting and Checking Numbers
console.log(23 === 23.0);
// Base 10 - 0 to 9. 1/10 = 0.1. 3/10 = 3.3333333
// Binary base 2 - 0 1
console.log(0.1 + 0.2);
console.log(0.1 + 0.2 === 0.3); // sad false
// Conversion
console.log(Number("23"));
console.log(+"23");
// Parsing
console.log(Number.parseInt("30px", 10));
console.log(Number.parseInt("e23", 10));
console.log(Number.parseInt(" 2.5rem "));
console.log(Number.parseFloat(" 2.5rem "));
// console.log(parseFloat(' 2.5rem '));
// Check if value is NaN
console.log(Number.isNaN(20));
console.log(Number.isNaN("20"));
console.log(Number.isNaN(+"20X"));
console.log(Number.isNaN(23 / 0));
// Checking if value is number
console.log(Number.isFinite(20));
console.log(Number.isFinite("20"));
console.log(Number.isFinite(+"20X"));
console.log(Number.isFinite(23 / 0));
console.log(Number.isInteger(23));
console.log(Number.isInteger(23.0));
console.log(Number.isInteger(23 / 0));
///////////////////////////////////////
// Math and Rounding (Math namespace)
console.log(Math.sqrt(25));
console.log(25 ** (1 / 2)); //root
console.log(8 ** (1 / 3));
console.log(Math.max(5, 18, 23, 11, 2));
console.log(Math.max(5, 18, "23", 11, 2)); // 23
console.log(Math.max(5, 18, "23px", 11, 2)); // NaN
console.log(Math.min(5, 18, 23, 11, 2)); //5
console.log(Math.PI * Number.parseFloat("10px") ** 2);
console.log(Math.trunc(Math.random() * 6) + 1);
/// generate in range
const randomInt = (min, max) =>
Math.floor(Math.random() * (max - min) + 1) + min;
// 0...1 -> 0...(max - min) -> min...max
// console.log(randomInt(10, 20));
// Rounding integers
console.log(Math.round(23.3));
console.log(Math.round(23.9));
console.log(Math.ceil(23.3));
console.log(Math.ceil(23.9));
console.log(Math.floor(23.3));
console.log(Math.floor("23.9"));
console.log(Math.trunc(23.3));
console.log(Math.trunc(-23.3));
console.log(Math.floor(-23.3));
// Rounding decimals
console.log((2.7).toFixed(0));
console.log((2.7).toFixed(3));
console.log((2.345).toFixed(2));
console.log(+(2.345).toFixed(2));
///////////////////////////////////////
// The Remainder Operator
console.log(5 % 2);
console.log(5 / 2); // 5 = 2 * 2 + 1
console.log(8 % 3);
console.log(8 / 3); // 8 = 2 * 3 + 2
console.log(6 % 2);
console.log(6 / 2);
console.log(7 % 2);
console.log(7 / 2);
const isEven = (n) => n % 2 === 0;
console.log(isEven(8));
console.log(isEven(23));
console.log(isEven(514));
labelBalance.addEventListener("click", function () {
[...document.querySelectorAll(".movements__row")].forEach(function (row, i) {
// 0, 2, 4, 6
if (i % 2 === 0) row.style.backgroundColor = "orangered";
// 0, 3, 6, 9
if (i % 3 === 0) row.style.backgroundColor = "blue";
});
});
///////////////////////////////////////
// Working with BigInt
console.log(2 ** 53 - 1); // max safe integer
console.log(Number.MAX_SAFE_INTEGER);
console.log(2 ** 53 + 1);
console.log(2 ** 53 + 2);
console.log(2 ** 53 + 3);
console.log(2 ** 53 + 4);
console.log(4838430248342043823408394839483204n);
console.log(BigInt(48384302)); // smll numbers
// Operations
console.log(10000n + 10000n);
console.log(36286372637263726376237263726372632n * 10000000n);
// console.log(Math.sqrt(16n));
const huge = 20289830237283728378237n;
const num = 23;
console.log(huge * BigInt(num));
// Exceptions
console.log(20n > 15);
console.log(20n === 20);
console.log(typeof 20n);
console.log(20n == "20");
console.log(huge + " is REALLY big!!!");
// Divisions
console.log(11n / 3n);
console.log(10 / 3);
///////////////////////////////////////
// Creating Dates
// Months are 0 based in JS
// Create a date
const now = new Date();
console.log(now);
console.log(new Date("Aug 02 2020 18:05:41"));
console.log(new Date("December 24, 2015"));
console.log(new Date(account1.movementsDates[0]));
console.log(new Date(2037, 10, 19, 15, 23, 5));
console.log(new Date(2037, 10, 31));
console.log(new Date(0));
console.log(new Date(3 * 24 * 60 * 60 * 1000));
// Working with dates
const future = new Date(2037, 10, 19, 15, 23);
console.log(future);
console.log(future.getFullYear());
console.log(future.getMonth());
console.log(future.getDate());
console.log(future.getDay());
console.log(future.getHours());
console.log(future.getMinutes());
console.log(future.getSeconds());
console.log(future.toISOString());
console.log(future.getTime());
console.log(new Date(2142256980000));
console.log(Date.now());
future.setFullYear(2040); //likewise month second
console.log(future);
- Precise Calculation -> Moment.js library
///////////////////////////////////////
// Operations With Dates
const future = new Date(2037, 10, 19, 15, 23);
console.log(+future);
const calcDaysPassed = (date1, date2) =>
Math.abs(date2 - date1) / (1000 * 60 * 60 * 24);
const days1 = calcDaysPassed(new Date(2037, 3, 4), new Date(2037, 3, 14));
console.log(days1);
- lingoes.net
///////////////////////////////////////
// Internationalizing Numbers (Intl)
const num = 3884764.23;
const options = {
style: "currency",
unit: "celsius",
currency: "EUR",
// useGrouping: false,
};
console.log("US: ", new Intl.NumberFormat("en-US", options).format(num));
console.log("Germany: ", new Intl.NumberFormat("de-DE", options).format(num));
console.log("Syria: ", new Intl.NumberFormat("ar-SY", options).format(num));
console.log(
navigator.language,
new Intl.NumberFormat(navigator.language, options).format(num)
);
/// DAte time format
//console.log("US: ", new Intl.DateTimeFormat("en-US", options).format(num));
const otheroptions = {
hour: "numeric",
minute: "numeric",
day: "numeric",
month: "long",
year: "numeric",
weekday: "short",
};
// how to get locale from browser
const locale = navigator.language;
///////////////////////////////////////
// Timers
// setTimeout
const ingredients = ["olives", "spinach"];
const pizzaTimer = setTimeout(
// ing1 = olives , ing2 = spinach
(ing1, ing2) => console.log(`Here is your pizza with ${ing1} and ${ing2} 🍕`),
3000,
...ingredients // arguements to callback
);
console.log("Waiting...");
if (ingredients.includes("spinach")) clearTimeout(pizzaTimer);
// setInterval
setInterval(function () {
const now = new Date();
console.log(now);
}, 1000);
// Selecting, Creating, and Deleting Elements
// Selecting elements
console.log(document.documentElement);
console.log(document.head);
console.log(document.body);
const header = document.querySelector(".header");
const allSections = document.querySelectorAll(".section");
console.log(allSections);
// this is not updated whenever you change
document.getElementById("section--1");
const allButtons = document.getElementsByTagName("button");
console.log(allButtons);
// these gets updated at that instance
// Try to delete one of elements and then check again allsections and allbuttons
console.log(document.getElementsByClassName("btn"));
// Creating and inserting elements
const message = document.createElement("div");
message.classList.add("cookie-message");
// for text only
message.textContent =
"We use cookied for improved functionality and analytics.";
// for html as well
message.innerHTML =
'We use cookied for improved functionality and analytics. <button class="btn btn--close-cookie">Got it!</button>';
// only one child can be there either of them
// header.prepend(message);
header.append(message);
// But if you want to have it both places
// header.append(message.cloneNode(true));
// Element to add before and after element
// header.before(message);
// header.after(message);
// .insertAdjacentHTML
// Delete elements
document
.querySelector(".btn--close-cookie")
.addEventListener("click", function () {
// message.remove(); //newer method
message.parentElement.removeChild(message); //older method
});
// Styles, Attributes and Classes
// Styles
message.style.backgroundColor = "#37383d";
message.style.width = "120%";
console.log(message.style.color);
console.log(message.style.backgroundColor);
// Access All styles of element
console.log(getComputedStyle(message).color);
console.log(getComputedStyle(message).height);
message.style.height =
Number.parseFloat(getComputedStyle(message).height, 10) + 30 + "px";
document.documentElement.style.setProperty("--color-primary", "orangered");
// Attributes
const logo = document.querySelector(".nav__logo");
console.log(logo.alt);
console.log(logo.className);
logo.alt = "Beautiful minimalist logo";
// Non-standard
console.log(logo.designer); // doesn't work
console.log(logo.getAttribute("designer")); // it works
logo.setAttribute("company", "Bankist");
console.log(logo.src); // different
console.log(logo.getAttribute("src")); //different
const link = document.querySelector(".nav__link--btn");
console.log(link.href);
console.log(link.getAttribute("href"));
// Data attributes
console.log(logo.dataset.versionNumber);
// it should be camelcase in js but while assigning to element in html it should be data-version-number
// Classes
logo.classList.add("c", "j");
logo.classList.remove("c", "j");
logo.classList.toggle("c");
logo.classList.contains("c"); // not includes
// Don't use
logo.clasName = "jonas";
const btnScrollTo = document.querySelector(".btn--scroll-to");
const section1 = document.querySelector("#section--1");
btnScrollTo.addEventListener("click", function (e) {
section1.scrollIntoView({ behavior: "smooth" });
});
///////////////////////////////////////
// Types of Events and Event Handlers
const h1 = document.querySelector("h1");
const alertH1 = function (e) {
alert("addEventListener: Great! You are reading the heading :D");
};
h1.addEventListener("mouseenter", alertH1);
setTimeout(() => h1.removeEventListener("mouseenter", alertH1), 3000);
// h1.onmouseenter = function (e) {
// alert('onmouseenter: Great! You are reading the heading :D');
// };
///////////////////////////////////////
// Event Propagation in Practice
const randomInt = (min, max) =>
Math.floor(Math.random() * (max - min + 1) + min);
const randomColor = () =>
`rgb(${randomInt(0, 255)},${randomInt(0, 255)},${randomInt(0, 255)})`;
document.querySelector(".nav__link").addEventListener("click", function (e) {
this.style.backgroundColor = randomColor();
console.log("LINK", e.target, e.currentTarget);
console.log(e.currentTarget === this);
// Stop propagation
// e.stopPropagation();
});
document.querySelector(".nav__links").addEventListener("click", function (e) {
this.style.backgroundColor = randomColor();
console.log("CONTAINER", e.target, e.currentTarget);
});
document.querySelector(".nav").addEventListener("click", function (e) {
this.style.backgroundColor = randomColor();
console.log("NAV", e.target, e.currentTarget);
});
// 1. Add event listener to common parent element
// 2. Determine what element originated the event
document.querySelector(".nav__links").addEventListener("click", function (e) {
e.preventDefault();
// Matching strategy
if (e.target.classList.contains("nav__link")) {
const id = e.target.getAttribute("href");
document.querySelector(id).scrollIntoView({ behavior: "smooth" });
}
});
///////////////////////////////////////
// DOM Traversing
const h1 = document.querySelector("h1");
// Going downwards: child
console.log(h1.querySelectorAll(".highlight"));
console.log(h1.childNodes);
console.log(h1.children);
h1.firstElementChild.style.color = "white";
h1.lastElementChild.style.color = "orangered";
// Going upwards: parents
console.log(h1.parentNode);
console.log(h1.parentElement);
h1.closest(".header").style.background = "var(--gradient-secondary)";
h1.closest("h1").style.background = "var(--gradient-primary)";
// Going sideways: siblings
console.log(h1.previousElementSibling);
console.log(h1.nextElementSibling);
console.log(h1.previousSibling);
console.log(h1.nextSibling);
console.log(h1.parentElement.children);
[...h1.parentElement.children].forEach(function (el) {
if (el !== h1) el.style.transform = "scale(0.5)";
});
///////////////////////////////////////
// Sticky navigation
const initialCoords = section1.getBoundingClientRect();
console.log(initialCoords);
window.addEventListener("scroll", function () {
console.log(window.scrollY);
if (window.scrollY > initialCoords.top) nav.classList.add("sticky");
else nav.classList.remove("sticky");
});
///////////////////////////////////////
// Sticky navigation: Intersection Observer API
const obsCallback = function (entries, observer) {
entries.forEach((entry) => {
console.log(entry);
});
};
const obsOptions = {
root: null,
threshold: [0, 0.2],
rootMargin: "-90px",
};
const observer = new IntersectionObserver(obsCallback, obsOptions);
observer.observe(section1);
const allSections = document.querySelectorAll(".section");
const revealSection = function (entries, observer) {
const [entry] = entries;
if (!entry.isIntersecting) return;
entry.target.classList.remove("section--hidden");
observer.unobserve(entry.target);
};
const sectionObserver = new IntersectionObserver(revealSection, {
root: null,
threshold: 0.15,
});
allSections.forEach(function (section) {
sectionObserver.observe(section);
section.classList.add("section--hidden");
});
// Lazy loading images
// filter blur in css
const imgTargets = document.querySelectorAll("img[data-src]");
const loadImg = function (entries, observer) {
const [entry] = entries;
if (!entry.isIntersecting) return;
// Replace src with data-src
entry.target.src = entry.target.dataset.src;
entry.target.addEventListener("load", function () {
entry.target.classList.remove("lazy-img");
});
observer.unobserve(entry.target);
};
const imgObserver = new IntersectionObserver(loadImg, {
root: null,
threshold: 0,
rootMargin: "200px",
});
imgTargets.forEach((img) => imgObserver.observe(img));
// Slider
const slider = function () {
const slides = document.querySelectorAll(".slide");
const btnLeft = document.querySelector(".slider__btn--left");
const btnRight = document.querySelector(".slider__btn--right");
const dotContainer = document.querySelector(".dots");
let curSlide = 0;
const maxSlide = slides.length;
// Functions
const createDots = function () {
slides.forEach(function (_, i) {
dotContainer.insertAdjacentHTML(
"beforeend",
`<button class="dots__dot" data-slide="${i}"></button>`
);
});
};
const activateDot = function (slide) {
document
.querySelectorAll(".dots__dot")
.forEach((dot) => dot.classList.remove("dots__dot--active"));
document
.querySelector(`.dots__dot[data-slide="${slide}"]`)
.classList.add("dots__dot--active");
};
const goToSlide = function (slide) {
slides.forEach(
(s, i) => (s.style.transform = `translateX(${100 * (i - slide)}%)`)
);
};
// Next slide
const nextSlide = function () {
if (curSlide === maxSlide - 1) {
curSlide = 0;
} else {
curSlide++;
}
goToSlide(curSlide);
activateDot(curSlide);
};
const prevSlide = function () {
if (curSlide === 0) {
curSlide = maxSlide - 1;
} else {
curSlide--;
}
goToSlide(curSlide);
activateDot(curSlide);
};
const init = function () {
goToSlide(0);
createDots();
activateDot(0);
};
init();
// Event handlers
btnRight.addEventListener("click", nextSlide);
btnLeft.addEventListener("click", prevSlide);
document.addEventListener("keydown", function (e) {
if (e.key === "ArrowLeft") prevSlide();
e.key === "ArrowRight" && nextSlide();
});
dotContainer.addEventListener("click", function (e) {
if (e.target.classList.contains("dots__dot")) {
const { slide } = e.target.dataset;
goToSlide(slide);
activateDot(slide);
}
});
};
slider();
///////////////////////////////////////
// Lifecycle DOM Events
document.addEventListener("DOMContentLoaded", function (e) {
console.log("HTML parsed and DOM tree built!", e);
});
window.addEventListener("load", function (e) {
console.log("Page fully loaded", e);
});
window.addEventListener("beforeunload", function (e) {
e.preventDefault();
console.log(e);
e.returnValue = "";
});
- Leaflet API Maps