Congratulations on your first successful chapter of Web Mapping! You now have created your first GitHub repo (Activity 1) and completed foundational tutorials for developing on the Open Web (Activity 2). Chapter 2 includes four lab lessons and ends with Activity 3, a debugging exercise to test your knowledge on JavaScript.
- In Lesson 1, we review the Document Object Model (DOM) and the ways that browsers interpret content on the web.
- In Lesson 2, we review JavaScript basics, including syntax rules for variables, functions, and the flow of execution. Lessons 1 and 2 are largely review from the Codecademy tutorials.
- In Lesson 3, we introduce jQuery, a helpful open-source library for efficiently writing JavaScript code.
- In Lesson 4, we discuss tips and tricks for debugging using the console.
After this chapter, you should be able to:
- Inspect the Document Object Model (DOM) in any browser
- Write variables, functions, and control structures in JavaScript
- Use jQuery to select and manipulate elements in the DOM
- Use the browser console to debug basic flaws in your code
Begin Chapter 2 by opening your unit-1 directory from Chapter 1. Begin by viewing the contents of the Chapter02 subdirectory, which includes two .js files. Then, navigate to the boilerplate subdirectory and select all of the contents. Copy these files and paste them into Chapter02. Move the original two .js files to the appropriate folder.
The Document Object Model, or DOM, is the standard tree structure, called the DOM tree, for organizing objects within a webpage (i.e., a "document"). The DOM is cross-platform and language-independent, meaning that it is not a specific language or technology, but a convention on how any language should structure a document. For instance, Figure 1.1 visualizes the DOM tree for the HTML boilerplate introduced in Chapter 1, showing how the DOM tree grows in complexity for even simple webpages.
The DOM includes all CSS styles and JavaScript variables, properties, and functions in addition to the HTML elements and their attributes. Importantly, JavaScript and other scripting languages reference objects through the DOM to enable interactivity on the web, including cartographic interaction. Accordingly, interactive web maps would be impossible without the DOM.
Thus, reading and revising code based on the DOM is a critical skill for efficient web development.
Every browser has a set of developer tools that support coding on the Open Web. The inspector tool identifies a given HTML element within the DOM structure.
You can activate the inspector on any webpage using right-/command-click and selecting "Inspect" or "Inspect Element" (the specific language varies by browser). Activating the inspector opens the inspector tab of the developer toolbox along the bottom or the side of the browser. Hovering over an HTML element in the inspector tab highlights the corresponding element in the webpage, also indicating the margin or padding associated with the element. You can click the right-facing arrow to open any element in the inspector tab to reveal the next nested level in the DOM.
If you click the box-arrow icon in the top-left of the inspector tab (a magnifying glass icon in some browsers), you then can hover over the webpage itself and identify the location of the highlighted HTML element in the DOM. Figure 1.2 shows the inspector tab in the Firefox browser, with the <body>
element selected.
You also can activate the inspector through the browser "Tools"/"Developer Tools" menu or the F12 keyboard shortcut.
The inspector primarily shows the HTML elements contained in the DOM and not the attributes, styles, and scripts also included in the DOM. Most browsers have a DOM tab (it will go by different names across browsers) that displays the entire DOM of a webpage, but the added complexity often is difficult to interpret for the purposes of debugging. We will return to additional developer tools in subsequent lessons that facilitate interpretation of other aspects of the DOM.
As introduced in Chapter 1, JavaScript is a web programming language for applying dynamic actions or behaviors to webpage content. In other words, JavaScript is the language of interaction on the web. In order to understand JavaScript, as any other scripting language, it is necessary to develop computational thinking skills, or the ability to think like a computer and work through the steps and processes taking place in a script. Journalist and interactive application developer Tasneem Raja compares computational thinking to cooking:
"Like a good algorithm, a good recipe follows some basic principles. Ingredients are listed first, so you can collect them before you start, and there's some logic in the way they are listed: olive oil before cumin because it goes in the pan first. Steps are presented in order, not a random jumble, with staggered tasks so that you're chopping veggies while waiting for water to boil. A good recipe spells out precisely what size of dice or temperature you're aiming for. It tells you to look for signs that things are working correctly at each stage—the custard should coat the back of a spoon. Opportunities for customization are marked—use twice the milk for a creamier texture—but if any in2gredients are absolutely crucial, the recipe makes sure you know it. If you need to do something over and over—add four eggs, one at a time, beating after each—those tasks are boiled down to one simple instruction."
Let's think about this analogy in terms of what you should know about JavaScript from the tutorial you completed:
- variables are the recipe ingredients
- functions are the cooking tasks
- syntax rules indicate how to precisely describe and measure each ingredient in the recipe
- console logging is like tasting the food throughout cooking to make small adjustments
- conditional statements indicate how to customize the recipe to different tastes
- loops indicate where to repeat a task
Additionally, comments indicate to human readers of the script—including you the programmer, as your memory often is imperfect—how the code works. You should add a comment to the line above any piece of the script that may be indiscernible to others. There is room for judgement about what parts of the script are obvious versus requiring clarification; as you code more, you will get a better idea of where you need to add reminders about how the script works. You also can use comments to temporarily disable parts of the script to isolate code blocks while debugging (more on this later).
Let's build a simple example to explore the logic of JavaScript (Example 2.1).
Reopen the index.html HTML boilerplate. Within the
<body>
of your index.html file, add a<div>
element with anid
attribute of"mydiv"
(do not forget to add a closing</div>
tag). Open the main.js file (also in your Chapter02 subdirectory) and add the Example 2.1 script. Make sure your main.js file is linked through a<script>
tag with asrc
attribute in index.html.
var mydiv = document.getElementById("mydiv");
mydiv.innerHTML = "Hello World";
In Example 2.1, you are accessing and manipulating mydiv
using JavaScript rather than hard-coding the div content using HTML.
It is best practice is to load any scripts after the rest of the HTML document loads so that the browser does not perform actions on elements that do not yet exist. To do this, we can use the window.onload
method to call a function containing our code (Example 2.2).
function myfunc(){
var mydiv = document.getElementById("mydiv");
mydiv.innerHTML = "Hello World.";
};
window.onload = myfunc();
Unlike other programming languages, JavaScript is not picky about the order of functions and function calls. Nonetheless, it is standard practice to call window.onload
last so the interpreter reads through everything once before reaching it.
Note that the code is neatly indented, similar to the nested structure of our HTML document. Everything within the myfunc()
function is indented an extra tab to show that it is contained by the function. Again, while indentation is not required, keeping your code tidy makes it much more readable and easier to debug. JavaScript interprets either a semicolon or a line break as the end of a statement, but it is good to get in the habit of using semicolons so that if you ever minify your code (compress it to a single, continuous line without comments), you will not have to go back through and add the semicolons.
JavaScript has a number of native methods for manipulating the DOM. Rather than introducing them all one-by-one, we slowly introduce these methods by example in the chapter lessons. Our focus instead is on following the computational logic when calling these methods. Carefully read through and consider the Example 2.3 script:
//initialize function called when the script loads
function initialize(){
cities();
};
//function to create a table with cities and their populations
function cities(){
//define two arrays for cities and population
var cities = [
'Madison',
'Milwaukee',
'Green Bay',
'Superior'
];
var population = [
233209,
594833,
104057,
27244
];
//create the table element
var table = document.createElement("table");
//create a header row
var headerRow = document.createElement("tr");
//add the "City" column
var cityHeader = document.createElement("th");
cityHeader.innerHTML = "City";
headerRow.appendChild(cityHeader);
//add the "Population" column
var popHeader = document.createElement("th");
popHeader.innerHTML = "Population";
headerRow.appendChild(popHeader);
//add the row to the table
table.appendChild(headerRow);
//loop to add a new row for each city
for (var i = 0; i < cities.length; i++){
var tr = document.createElement("tr");
var city = document.createElement("td");
city.innerHTML = cities[i];
tr.appendChild(city);
var pop = document.createElement("td");
pop.innerHTML = population[i];
tr.appendChild(pop);
table.appendChild(tr);
};
//add the table to the div in index.html
var mydiv = document.getElementById("mydiv");
mydiv.appendChild(table);
};
//call the initialize function when the window has loaded
window.onload = initialize();
If you add this code into main.js and reload index.html in the browser using Prepros, you should now see a neat little table on the webpage and in the DOM (Figure 2.2)
JavaScript provides multiple ways of implementing most computational tasks, a concept we describe in lecture as interface flexibility. As a review of the Activity 2 Codecademy tutorials, let's examine several alternatives to the above example.
Example 2.3 uses two arrays cities
and population
to hold the city names versus population, with a looping structure then populating the two-dimensional table with this content. Alternatively, we could use one array cityPop
of JavaScript objects to hold the table content instead of two separate arrays (Example 2.4).
//Example 2.3 line 6...function to create a table with cities and their populations
function cities(){
//define an array of objects for cities and population
var cityPop = [
{
city: 'Madison',
population: 233209
},
{
city: 'Milwaukee',
population: 594833
},
{
city: 'Green Bay',
population: 104057
},
{
city: 'Superior',
population: 27244
}
];
//...
//Example 2.3 line 41...loop to add a new row for each city
for (var i = 0; i < cityPop.length; i++){
var tr = document.createElement("tr");
var city = document.createElement("td");
city.innerHTML = cityPop[i].city; //NOTE DIFFERENT SYNTAX
tr.appendChild(city);
var pop = document.createElement("td");
pop.innerHTML = cityPop[i].population; //NOTE DIFFERENT SYNTAX
tr.appendChild(pop);
table.appendChild(tr);
};
This format has the advantage of explicitly associating each city with its population as two properties of the same object, rather than simply relying on the order of elements in the arrays to associate them, which easily could have an off-by-one error with a missing or misplaced comma, or otherwise omitted value.
You also can create the array of objects using longform syntax, defining each object individually and then pushing it into the cityPop
array (Example 2.5):
//Example 2.3 line 8...create an empty array
var cityPop = [];
//create the first city object
var madison = {};
//add each property to the object
madison.city = 'Madison';
madison.population = 233209;
//push the city object into the array
cityPop.push(madison);
//repeat...
var milwaukee = {};
milwaukee.city = 'Milwaukee';
milwaukee.population = 594833;
cityPop.push(milwaukee);
var greenBay = {};
greenBay.city = 'Green Bay';
greenBay.population = 104057;
cityPop.push(greenBay);
var superior = {};
superior.city = 'Superior';
superior.population = 27244;
cityPop.push(superior);
When would you use this longform syntax? It is not necessary here, but you might imagine a case where you wanted to dynamically populate an array—that is, add objects to it at different times in the program. You might also want to add a new property or change some property of an object some time after the object was created.
Keep in mind that JavaScript is case sensitive, so consistent capitalization is important! If you define a variable as superior
and then call it as Superior
, the interpreter will not recognize the variable when you call it. It is standard practice to write variable names using camelCase, or a lowercase first letter and uppercase for any subsequent letter that begins a separate word (for instance, greenBay
). You cannot use spaces in variable names. You can include spaces for a string object (any collection of characters surrounded by single or double quotes, such as 'Green Bay'
), although special character (such as a quotation mark) require a leading escape \
character. Numerical values are assigned to variables without using quotes (quotes would turn them into strings) and cannot be interrupted by any non-numerical characters (such as a comma), except for a single period, which is used as a decimal point.
Example 2.3 uses a for
loop to create the table. Alternatively, you can use a while
loop, a for each
loop, or nested loops (Example 2.6):
//WHILE LOOP...Example 2.4 line 25
//define a counter variable
var i = 0;
//start the loop
while (i < cityPop.length){
var tr = document.createElement("tr");
var city = document.createElement("td");
city.innerHTML = cityPop[i].city;
tr.appendChild(city);
var pop = document.createElement("td");
pop.innerHTML = cityPop[i].population;
tr.appendChild(pop);
table.appendChild(tr);
//increment counter
i++;
};
//FOREACH LOOP...Example 2.4 line 25
cityPop.forEach(function(cityObject){
var tr = document.createElement("tr");
var city = document.createElement("td");
city.innerHTML = cityObject.city; //NOTE DIFFERENT SYNTAX
tr.appendChild(city);
var pop = document.createElement("td");
pop.innerHTML = cityObject.population; //NOTE DIFFERENT SYNTAX
tr.appendChild(pop);
table.appendChild(tr);
});
//FOREACH LOOP WITH OBJECT FOR LOOP...Example 2.4 line 25
cityPop.forEach(function(cityObject){
var tr = document.createElement("tr");
for (var property in cityObject){
var td = document.createElement("td");
td.innerHTML = cityObject[property];
tr.appendChild(td);
};
table.appendChild(tr);
});
With the while
loop, make sure to define the counter variable first, outside of the loop, and increment the counter within the loop. Otherwise, you will be stuck with an infinite loop!
The forEach
loop uses an anonymous function to perform the task for each element in the cityPop
array, passing the element into the function as a variable (in this case, cityObject
). Anonymous functions do not include a function name and commonly are used in JavaScript when the function is not needed after its initial creation. Note that the forEach
method must be closed by both a curly brace (for the anonymous function) and a parenthesis (for the forEach
method itself).
The third looping structure might be the most challenging to wrap your head around. It is actually a set of nested loops, with a second for
loop inside of the forEach
loop. The inner for
loop iterates through each property of each object in the array (i.e., cityObject.city, cityObject.population), filling a cell with its value. The variable property
holds the name of the property (i.e., 'city', 'population'). The loop uses brackets to get the value of the variable property
and use it to call the value of that property in the object. Thus, cityObject[property]
is equivalent to cityObject.city
and cityObject.population
, each in turn during the inner for
loop.
Finally, let's return to the original loop structure to review some basic conditional statements (Example 2.7).
//Example 2.4 line 25...loop to add a new row for each city
for (var i = 0; i < cityPop.length; i++){
var tr = document.createElement("tr");
var city = document.createElement("td");
//first conditional block
if (cityPop[i].city == 'Madison'){
city.innerHTML = 'Badgerville';
} else if (cityPop[i].city == 'Green Bay'){
city.innerHTML = 'Packerville';
} else {
city.innerHTML = cityPop[i].city;
}
tr.appendChild(city);
var pop = document.createElement("td");
//second conditional block
if (cityPop[i].population < 500000){
pop.innerHTML = cityPop[i].population;
} else {
pop.innerHTML = 'Too big!';
};
tr.appendChild(pop);
table.appendChild(tr);
};
The Example 2.7 script presents two conditional blocks. The first looks at the name of the city, giving a different value to the table cells for Madison and Green Bay. Remember that in JavaScript (as in most programming languages), conditional equality is denoted by ==
(equal value) or ===
(equal value and type), while =
denotes assignment of a value to a variable. Thus, make sure you are using the correct number of equals signs in your comparison statements and not accidentally reassigning the values of your variables.
The second conditional block uses a <
to compare the population value to a threshold, above which the value 'Too big!' is printed in the cell instead of the population. Any conditional that can be reduced to true or false can be represented in one line, shorthand syntax (Example 2.8):
//Example 2.7 line 19
pop.innerHTML = cityPop[i].population < 500000 ? cityPop[i].population : 'Too big!';
As the question mark implies, you can think of this statement as asking a question: is the population less than 500,000? If it is, assign cityPop[i].population
to pop.innerHTML
. If not, assign 'Too big!'
to pop.innerHTML
. We recommend starting with the longform conditional syntax, as it is easy to invert the order of conditions with the shorthand syntax.
As you can see from the previous lesson, you need an awful lot of JavaScript code to make just one tiny table! While JavaScript works great for computational logic, it is a bit clunky when picking HTML elements out of the DOM and manipulating them. This is where jQuery comes in handy.
jQuery is an open source library that simplifies a number of common applications of JavaScript for web development, including DOM traversal and manipulation, event handling, cross-browser consistency, and AJAX. Our focus this chapter is on jQuery for interacting with the DOM; we will cover additional features in future chapters. While alternatives are emerging, jQuery remains one of the most widely-used open-source coding libraries and thus is worth learning.
The first step to using jQuery is to add it to the lib folder of your website directory.
From the jQuery website, download the latest version of jQuery (uncompressed, development version; example code may reflect older versions). Save this js file to the lib folder in your website directory.
After adding the jQuery.js file to your website directory, link to it in the <body>
of your index.html file (Example 3.1).
<script type="text/javascript" src="lib/jquery-3.5.1.js"></script>
Make sure you add this script link above the link to your main.js file in index.html, as linked files are read by the browser's interpreter in the order in which they are linked. Otherwise, the code in main.js will not be able to find the jQuery library!
Because jQuery is an open-source library, you can open it up and read through it. The jQuery developers made it especially easy to read the development version by using good formatting and heavily commenting throughout, although we do not expect you to understand all of the script for this workbook.
The jQuery library essentially is a series of functions that support common tasks. When a function is part of a library, it is generally referred to as a method of that library. jQuery is somewhat unique in that it supports method chaining, or the calling of multiple methods in sequence using dot syntax, resulting in simplified code. Example 3.2 shows a two method chain using jQuery.
jQuery('#mydiv').html('Hello World');
The first part of the Example 3.2 method chain is jQuery('#mydiv')
. This method selects the HTML <div>
with the id attribute 'mydiv'
and returns it to the script for manipulation. The second method, html('Hello World')
, writes the text 'Hello World' to the <div>
. The method chain above is functionally equivalent to Example 2.1 above.
You already can start to see why jQuery is so useful. Rather than having to designate a variable to hold the <div>
element for processing and use the lengthy document.getElementById
method, jQuery('#mydiv')
grabs the element and prepares it for manipulation by other jQuery methods right away.
jQuery also provides alias $
syntax for its name to further reduce the size of your code (Example 3.3).
$('#mydiv').html('Hello World');
jQuery uses a selector to find the element or elements you want to manipulate. In the case above, we gave our <div>
an id
attribute, which we can access using pound #
sign, just like in CSS. If we had assigned a class
attribute to the <div>
, we can access it with a .
, as in '.mydiv'
. You also can select all elements of a particular tag in the document by using just the tag name with no prefix character, such as $('div')
to select all div elements.
Note that the selector must be in quotes (single or double) unless it is a variable that holds the string value of the attribute used for the selection, or it is a keyword recognized by jQuery (document
or window
). It is good to get in the habit of naming any HTML elements with id
and/or class
attributes that you subsequently might want to manipulate in your script or your CSS stylesheet. Keep in mind that each unique id
value should be used for only one element in the document, whereas class
is meant to be used to give the same identifier to multiple elements.
Knowing how jQuery syntax works, you now can rewrite the table script using jQuery's .append()
method, which adds elements to a webpage (Example 3.4):
//initialize function called when the script loads
function initialize(){
cities();
};
//function to create a table with cities and their populations
function cities(){
//define two arrays for cities and population
var cityPop = [
{
city: 'Madison',
population: 233209
},
{
city: 'Milwaukee',
population: 594833
},
{
city: 'Green Bay',
population: 104057
},
{
city: 'Superior',
population: 27244
}
];
//append the table element to the div
$("#mydiv").append("<table>");
//append a header row to the table
$("table").append("<tr>");
//add the "City" and "Population" columns to the header row
$("tr").append("<th>City</th><th>Population</th>");
//loop to add a new row for each city
for (var i = 0; i < cityPop.length; i++){
//assign longer html strings to a variable
var rowHtml = "<tr><td>" + cityPop[i].city + "</td><td>" + cityPop[i].population + "</td></tr>";
//add the row's html string to the table
$("table").append(rowHtml);
};
};
//call the initialize function when the document has loaded
$(document).ready(initialize);
Compare the (much shorter) jQuery script in Example 3.4 to our original script in Example 2.3. Note that the strategy for adding elements to the webpage has changed. In the original Example 2.3 script, we first created each element, then added the content to the element, and finally—only when the element was complete—added it to the DOM. Thus, the <table>
—the outermost element—was the very first element created, but the last to be appended, and the only element appended to an existing element in the DOM (the <div>
).
jQuery enables a different approach: HTML elements are created at the same time they are added to the DOM. The <table>
element still is the first element we manipulate, but it is immediately appended to the existing <div>
. Notice there is no closing </table>
tag; you could provide one, but jQuery adds this automatically for elements that need it if you do not. Once appended, it then is possible for jQuery to select the table out of the DOM for manipulation.
Continuing with the Example 3.4 code, we append the first table row element <tr>
directly to the <table>
, and to that we append the full html string of the header cells. This is just like writing HTML code within index.html, except that it is contained within a string and you need not worry about the formatting. Within the loop, we again append an HTML string, this time representing the full row. Note that we could have also done this for the header row, combining the previous two .append()
methods. Instead of writing our HTML string just as a parameter of the .append()
method, we first assign it to the variable rowHtml
. This keeps the code tidier by avoiding a line of script so long that it bleeds off the page (although sometimes this is simply unavoidable). We form the HTML string by concatenating strings of the proper HTML tags with the cityPop[i].city
and cityPop[i].population
properties, looping through the cityPop
array until there are no more objects to append as rows to the table.
At the bottom of the script, we replace window.onload = intialize()
with $(document).ready(initialize)
. This will execute the script as soon as the DOM is prepared, before all images and frames are loaded, making the loading of the site faster.
When writing HTML strings, make sure that you add all of the proper closing tags to your elements in the correct order. Unclosed elements may result in a DOM structure you did not anticipate, with odd formatting results. Figure 3.1 visualizes how the HTML elements in your table are organized in the DOM tree.
As you develop your lab projects, you will use many more jQuery methods. All of jQuery's methods are described in the jQuery documentation. Here we cover several of the more useful jQuery methods for manipulating the DOM. Each of these methods is called on a jQuery selection using dot syntax.
.attr(): This method allows you to get or set an attribute of the selected HTML element. It takes one or two parameters: the first parameter is the name of the attribute (e.g., 'id'
or 'class'
), while the second, optional parameter is a value with which to replace that attribute's current value (Example 3.5).
//Added at Example 3.5 line 44...
//get the div id
var theid = $('#mydiv').attr('id');
//theid is 'mydiv'; add it as text to the div
$('#mydiv').append(theid);
//add the class 'foo' to the div
$('#mydiv').attr('class', 'foo');
//Check your work with the Inspector!
.css(): This method gets or sets the value of the style
attribute of the selected element(s) using CSS styles. It is useful for changing element styles dynamically. It will, however, override any styles in your style.css file that apply to the element(s), as in-line styles (those designated through the HTML element style
attribute) always override styles stored in separate stylesheets. You can either give it the name and value of one style as strings, or change multiple styles at once as an object (Example 3.6).
//Added below Example 3.6...
//change the text color
$('#mydiv').css('color', 'red');
//change the text size and alignment
$('#mydiv').css({
'font-size': '2em',
'text-align': 'left'
});
//get the text color and add it as text to the div
var thecolor = $('#mydiv').css('color');
$('#mydiv').append(thecolor);
//fooled ya! thecolor is rgb(255, 0, 0), the CSS interpreter's translation of the keyword 'red'
.each(): This method accepts as its parameter a function that loops through all of the selected elements. This is useful if you need to use individual attribute values of each element or need to assign different values to each element using imported data (Example 3.7).
//Added below Example 3.7...
//iterate over each script element and add each one's source url as text to the div
$('script').each(function(){
var thesource = $(this).attr('src');
$('#mydiv').append(thesource);
});
Note that $(this)
inside of the loop selects the current element. You need not use .each()
if you are assigning the exact same value to a set of selected elements; for that, simply chain the method that does what you want to the selection.
.on() and .off(): These are jQuery's standard event listener methods that apply actions or behaviors based on user- or system-driven events, similar to JavaScript's native addEventListener()
and removeEventListener()
methods. .on()
usually takes two parameters: the name of the event to be listened for (such as 'click'
, 'mouseover'
, 'mouseout'
, etc.), and an event handler function that executes when the event occurs (Example 3.8). .off()
predictably does the opposite of .on()
, although you need to provide exactly the same event name and handler function to actually remove the event listener from the element. Thus, while you can provide either method an anonymous function as the second parameter, if you wish to remove the listener it is best to define this function separately and pass the function name to the .on()
and .off()
methods. There also are several alias methods available for popular event listeners.
//Added below Example 3.8...
//click listener with anonymous handler function
$('table').on('click', function(){
alert('Madison Rocks! Go Badgers!');
});
//alias method for the click event listener
$('table').click(function(){
alert('Visit Superior and see the big lake!');
});
//named handler function for removable listener
function clickme(){
alert('Yeah Green Bay! Go Packers!');
};
//add the event listener
$('table').on('click', clickme);
//remove the event listener
$('table').off('click', clickme);
Note: Try functions above one by one, or the last .off()
function will turn off all event listeners.
Despite how hackers are depicted in Hollywood movies, you will spend most of your time debugging code rather than writing code. Jeffrey Elkner, Alan B. Downey, and Chris Meyers try to put debugging in perspective:
"One of the most important skills you will acquire is debugging. Although it can be frustrating, debugging is one of the most intellectually rich, challenging, and interesting parts of programming... In some ways, debugging is like detective work. You are confronted with clues, and you have to infer the processes and events that led to the results you see.
"Debugging is also like an experimental science. Once you have an idea what is going wrong, you modify your program and try again. If your hypothesis was correct, then you can predict the result of the modification, and you take a step closer to a working program. If your hypothesis was wrong, you have to come up with a new one. As Sherlock Holmes pointed out, When you have eliminated the impossible, whatever remains, however improbable, must be the truth."
Or, in the words of the venerable Douglas Adams:
If you have been testing each code example in this chapter's lessons, you probably have run into some unexpected problems. Maybe you accidentally misplaced some punctuation, misspelled or incorrectly capitalized a word, or forgot an essential character. Maybe you copied the code from each example into main.js without removing or commenting out the original Example 2.3 code the new example was intended to replace. Maybe you forgot to update your link to jQuery to the lib folder versus the js folder in index.html. Maybe you forgot to preview your changes with Prepros or to clear your cache between refreshes. The list of possible issues goes on and on...
Such difficulties are normal, everyday experiences for every level of programmer, beginner to expert. The trick to overcoming routine errors is learning how to efficiently debug your code as you go. A single wrong character in the script can break your entire webpage, and it requires some sleuthing to find the pointy needle in the haystack. Fortunately, every browser's developer toolset comes with a vital tool that allows us to do just that: the console.
The console, usually accessed by a tab or button next to the inspector, is where errors and other messages from the scripts are printed. Often, it actually is more convenient for you the developer if your console shows an error, because the console error should tell you exactly where in the script the error is occurring. For instance, say your script failed and you found a syntax error in the console (Figure 4.1).
The error statement identifies line 25 of your main.js file, where there may be a missing curly brace. Now the problem should be easier to spot in the buggy Example 4.1 code. The object representing the City of Superior requires a closing curly brace before the bracket closes the overall array.
//Example 3.5 line 6...
function cities(){
//define two arrays for cities and population
var cityPop = [
{
city: 'Madison',
population: 233209
},
{
city: 'Milwaukee',
population: 594833
},
{
city: 'Green Bay',
population: 104057
},
{
city: 'Superior',
population: 27244
]; //this is line 25 of Example 3.4
Similarly, perhaps you have an undefined variable that you try to manipulate through JavaScript in the code. The console will display a reference error, noting the particular variable (Figure 4.2).
There are multiple reasons a variable could be undefined, but a very common cause is inconsistent capitalization. The error above happens because in one place the script uses citypop
and in another it uses cityPop
. Since the variable was defined as cityPop
, citypop
is not recognized.
Most errors in the console are explanatory and helpful, but occasionally you will run into an error that is mysterious. For example, while developing the example script we have been using, I ran into an error I had never seen before (Figure 4.3):
"TypeError: context is null"? What the what? The Console was telling me that the problem was in jQuery, but rarely does a publicly-distributed code library come with bugs in it. The problem instead was what I was feeding to jQuery through my script, not something in jQuery itself. But annoyingly, the console cannot discriminate where in your script the problem is if the error is thrown by the code library. The way I addressed this issue was to comment out each jQuery method chain in my script in turn until I found the line that was the culprit. This one was relatively easy to find; it turns out the problem was this statement that I was trying to use document
as a selector for the jQuery append()
method (Example 4.2).
$(document).append("<table>");
jQuery only accepts document
as a selector for its .ready()
method, not for .append()
. The latter method requires the selection of an existing HTML element. I thus changed the selector to a div reference by id "#mydiv"
(Example 4.3).
$("#mydiv").append("<table>");
This update resolved the issue.
What about an error that causes your script to fail silently? For instance, while I was tinkering with the example script for this lesson, I was able to generate a table with just the headers and no errors in the console (Figure 4.4):
Example 4.4 shows my original script producing this silent error.
//initialize function called when the script loads
function initialize(){
cities();
};
//function to create a table with cities and their populations
function cities(){
//define two arrays for cities and population
var cityPop = [
{
city: 'Madison',
population: 233209
},
{
city: 'Milwaukee',
population: 594833
},
{
city: 'Green Bay',
population: 104057
},
{
city: 'Superior',
population: 27244
}
];
//append the table element to the div
$("#mydiv").append("<table>");
//append a header row to the table
$("table").append("<tr>");
//add the "City" and "Population" columns to the header row
$("tr").append("<th>City</th><th>Population</th>");
//loop to add a new row for each city
for (var i = 0; i < cities.length; i++){
//assign longer html strings to a variable
var rowHtml = "<tr><td>" + cityPop[i].city + "</td><td>" + cityPop[i].population + "</td></tr>";
//add the row's html string to the table
$("table").append(rowHtml);
};
};
//call the initialize function when the window has loaded
$(document).ready(initialize);
Can you see the problem? I did not. I could tell from preview that my header row was drawing correctly, but it was as though my loop did not exist. There were a few things that could be going on, so I needed to use a process of elimination to test for different issues. This is where console.log
comes in handy. This native JavaScript method prints whatever you want to the console, allowing you to make visible what is going on in the script.
Let's debug! First, we can see whether the script is stopping for some reason before it reaches the loop. To do this, we can add a simple console.log
statement just before the loop to see if it will execute (Example 4.5):
//Example 4.4 line 36...
console.log("Hello World");
//loop to add a new row for each city
for (var i = 0; i < cities.length; i++){
//assign longer html strings to a variable
var rowHtml = "<tr><td>" + cityPop[i].city + "</td><td>" + cityPop[i].population + "</td></tr>";
//add the row's html string to the table
$("table").append(rowHtml);
};
The results (Figure 4.5):
Next, is the problem with the code in the loop or the loop itself? To see if the loop is executing, let's move our console.log
statement to the first order of business within the loop (Example 4.6):
//Example 4.4 line 36...
//loop to add a new row for each city
for (var i = 0; i < cities.length; i++){
console.log("Hello World");
//assign longer html strings to a variable
var rowHtml = "<tr><td>" + cityPop[i].city + "</td><td>" + cityPop[i].population + "</td></tr>";
//add the row's html string to the table
$("table").append(rowHtml);
};
Result (Figure 4.6):
Aha! The loop is not executing at all. So let's look at the opening statement of our loop (Example 4.7):
for (var i = 0; i < cities.length; i++){
Look carefully at every character in the line for errors with the syntax. There are none (plus the console didn't show a syntax error), so let's look at the variables. The only variable that is not defined within the statement is cities
. Let's check this variable with a console.log
statement just above the loop (Example 4.8).
console.log(cities);
//loop to add a new row for each city
for (var i = 0; i < cities.length; i++){
Result (Figure 4.7):
Notice in the Console that cities
is a function, not a variable! Thus it is defined, but has no length
property. If you click the button next to the function name in the console, you can see where in the script the function is defined (Figure 4.8).
Why then did I use cities
in my loop? Remember back to our very first script, when we were using two arrays for our data instead of an array of objects (Example 4.9).
//THE OLD DATA...Example 4.4 line 6
function cities(){
//define two arrays for cities and population
var cities = [
'Madison',
'Milwaukee',
'Green Bay',
'Superior'
];
var population = [
233209,
594833,
104057,
27244
];
//THE NEW DATA...Example 4.4 line 6
function cities(){
//define two arrays for cities and population
var cityPop = [
{
city: 'Madison',
population: 233209
},
{
city: 'Milwaukee',
population: 594833
},
{
city: 'Green Bay',
population: 104057
},
{
city: 'Superior',
population: 27244
}
];
I simply forgot to change the name of the variable cities
to cityPop
in the loop when I changed the structure of my data. Since both the original cities
variable and the new cityPop
variable hold arrays of the same length (4, the number of cities in our data), they should work the same way in a loop statement. We can now fix the loop statement (Example 4.10):
//cities CHANGED TO cityPop...Example 4.4 line 36
for (var i = 0; i < cityPop.length; i++){
//assign longer html strings to a variable
var rowHtml = "<tr><td>" + cityPop[i].city + "</td><td>" + cityPop[i].population + "</td></tr>";
//add the row's html string to the table
$("table").append(rowHtml);
};
This might seem like a simple error, but much of debugging involves tediously tracking down simple errors such as this by testing various parts of the script with console.log
statements. A smart strategy is to add temporary console.log
statements to check your work every time you write a new piece of code. This way, you know if something is wrong right away and have less script to debug if any problems arise.
- View the main_with_debug.js script found in the js folder of the Chapter02 subdirectory within your unit-1 repo. Replace the script link in index.html from main.js to main_with_debug.js.
- Debug the main_with_debug.js script to get the table to draw properly with the hover and click interactions (see preview below).
- Add comments to main_with_debug.js explaining what the script is doing at each step.
- Submit your working folder as a zip file (including HTML, JavaScript file, etc.)
This work is licensed under a Creative Commons Attribution 4.0 International License.
For more information, please contact Robert E. Roth ([email protected]).