layout | title | subtitle | bigimg |
---|---|---|---|
post |
Deadly sins of javascript implementation |
Javascript bad practices |
/img/js-sins.jpg |
Global variables and function defined in one script can be overwritten by second script if it has same names. This might cause unexpected results. Therefore, we should be careful while using global variables or functions or event handlers.
- Every script in a HTML document have same right as other scripts, therefore code snippet written in one script can be overwritten by other script.
- If you define global variables or function names and include other file using same names, than second file will overwrite the first implementation.
- If you attach event on onEvent DOM level-1 function and include another file which uses same event name, than second file will overwrite the first implementation.
Let's say you have script_one.js file with following code
x = 4;
function init() {
alert("init script one");
document.getElementsByTagName('h1')[0].onclick = function() {
this.style.background = 'blue';
}
}
alert("The value of x in script one is "+ x);
window.onload = init;
Immediately after script_one.js, include script_two.js with following code block
x = 40;
function init() {
alert("init script two");
document.getElementsByTagName('h1')[0].onclick = function() {
this.style.background = 'red';
}
}
alert("The value of x in script two is "+ x);
window.onload = init;
If you execute these scripts in a browser, you will find that it alerts 4 and 40 and executes init script two but not script one. It means init function of script_two.js overwrites the first init and init function of script_one.js is never executed.
Background of the text becomes red if you click on the text but it never gets blue. Demo
Solution to avoid this problem is to avoid using onEvent handlers.You can use proper DOM level-2 event handlers (it won't work in IE) as described below. Similarly, wrap your code with unique function names to avoid overriding from each other.
var scriptOne = function() {
var x = 5;
function init() {
alert('script one init');
document.getElementsByTagName('h1')[0].addEventListener(
'click',
function(e) {
var t = e.target;
t.style.background = 'blue';
},
false
);
}
alert('x inside is '+x);
return {init:init};
}();
window.addEventListener('load',scriptOne.init,false);
alert('x outside is '+x);
var scriptTwo = function() {
var x = 10;
function init() {
alert('script two init');
document.getElementsByTagName('h1')[0].addEventListener(
'click',
function(e){
var t = e.target;
t.style.color = 'white';
},
false
);
}
alert('x inside is '+x);
return {init:init};
}();
window.addEventListener('load',scriptTwo.init,false);
alert('x outside is '+x);
In this way everything executes as expected, x inside is 5 and than 10 and text changes to blue when you click on the text.
By moving x inside a function and using keyword var infront of them, we have made them visible within those functions and restricted outside world to use that variable directly. This is called Closure which is the powerful feature of javascript.
Module Pattern is considered as the best practices while creating closure function that retruns function which should be acceble to the outside world.
var scriptOne = function() {
var x = 4;
var f = 3;
function load(){}
function init(){}
return {
init:init
}
}();
In this way, we can access load and init function by scriptOne.load() or scriptOne.init(). This technique will safeguard all your variables and functions.
Most of the time we are habituated to hack our own code and get the things done. But this might lead to unmaintanable code which is hard for other future developers to work on. Therefore, its necessary to follow up some standards.
- Regardless of the environment it will be displayed in, only essential content and mark-up should be in HTML.
- Any sort of Look And Feel elements should be controlled by CSS. Try to avoid changing color stuff using javascript.
- Any sort of interactions with the user beyond hover effects should be done by javascript.
Let's take an example of how things can be done in a hacky and standard way. Let's say we have this HTML elements
<h2>Section 1</h2>
<div class="section">
<p>Section 1 content</p>
</div>
<h2>Section 2</h2>
<div class="section">
<p>Section 2 content</p>
</div>
<h2>Section 3</h2>
<div class="section">
<p>Section 3 content</p>
</div>
<h2>Section 4</h2>
<div class="section">
<p>Section 4 content</p>
</div>
Now we want to open each section when its heading is clicked and highlight the content section. You can play around in plunker. Normal jQuery solution to toggle the clicked section would be
$(document).ready(function() {
$('.section').hide();
$('h2').click(function(e) {
$(this).next().toggle();
});
});
To highlight the content of the current section we normally prefer this way i.e controlling styling through javascript.
$(document).ready(function(){
$('.section').hide();
$('h2').click(function(e){
$(this).next().toggle();
$(this).next().css('background','#ccc');
$(this).next().css('border','1px solid #999');
$(this).next().css('padding','5px');
})
});
Few things are wrong with this apporach.
- For begineer we have made it hard to maintain the code by controlling look and feel through javascript but not CSS.
- Performance issue: eventhough jQuery is fast enough, there is a lot of code behind $('.section').hide().
- We have asked jQuery to find the next siblings four times and set the CSS. We can either set the value of $(this).next() in a variable and use it but we can avoid it by using map as below
$(document).ready(function(){
$('.section').hide();
$('h2').click(function(e){
$(this).next().toggle().css({
'background':'#ffc',
'border':'1px solid #999',
'padding':'5px'
});
})
});
In this case we can open all the sections when heading for each section is clicked. What if we want to open only current section which is clicked. Normally, we would prefer following approach.
$(document).ready(function(){
$('.section').hide();
$('h2').click(function(e){
$('.section').hide();
$(this).next().toggle().css({
'background':'#ffc',
'border':'1px solid #999',
'padding':'5px'
});
})
});
This appraoch works fine but we are looping through the document lots of time whenever heading is clicked which is slow. Whenever $('.section').hide() is called after each click, jQuery goes through the document and applies the actions on the particular element which will create performance issue if it is called multiple times. It can be optimized by using current variable and avoid looping though the DOM.
$(document).ready(function(){
var current = false;
$('.section').hide();
$('h2').click(function(e){
if(current){
current.hide();
}
current = $(this).next();
current.toggle().css({
'background':'#ffc',
'border':'1px solid #999',
'padding':'5px'
});
})
});
All the methods explained above uses javascript to control styling which is not the best approach. We can use CSS along with JS to achieve this functionality.
$(document).ready(function(){
$('body').addClass('js');
var current = null;
$('h2').click(function(e){
if(current){
current.removeClass('current');
}
current = $(this).next().addClass('current');
})
});
<style type="text/css" media="screen">
.section{
border:1px solid #999;
background:#ccc;
}
.js .section{
display:none;
}
.js .current{
display:block;
border:1px solid #999;
background:#ffc;
}
</style>
By adding the class js to the body of the document and toggling the class current for the current section, we maintained control of the look and feel in CSS. With this apporach the handle will be reusable by designer and maintainer. Anything without the .js selector would be the non-scripting-enabled version of a part of the document, and anything with the .js selector is applied only when JavaScript is available.
var x = 2;
for(var x = 5; x <= 10; x++) {
console.log("Value of x inside for loop: ", x);
}
console.log("Value of x outside for loop: ",x);
//Executing above code logs following output
Value of x inside for loop: 5
Value of x inside for loop: 6
Value of x inside for loop: 7
Value of x inside for loop: 8
Value of x inside for loop: 9
Value of x inside for loop: 10
Value of x outside for loop: 11
We would expect that the varaible x used in the for-loop is visible only inside the for-loop and value of x outside would still be 2 but that's wrong. Actually, while executing this code interpreter first define a value of a variable x as undefined and than it defines 2 and later 5 to 10. Therefore, value of the x outside the for-loop is 11.