-
-
Notifications
You must be signed in to change notification settings - Fork 354
JavaScript Coding Standards
The core feature set of Rock's front-end functionality depend heavily upon a few common JavaScript libraries. These will be included on every page:
- jQuery
- jQuery UI (may be subject to change)
- Bootstrap
- Kendo UI
- jQuery TinyScrollbar
- Bootbox
We seek to minimize dependence on third party libraries. We will be maintaining these and are committing to staying on a recent version of each library. In the interest of performance, all of these libraries will be bundled together and minified, and included in the DOM via ~/bundles/RockLibs
.
bundles.Add( new ScriptBundle( "~/bundles/RockLibs" ).Include(
"~/Scripts/jquery.js",
"~/Scripts/jquery-ui-1.10.0.custom.min.js",
"~/Scripts/Kendo/kendo.web.min.js",
"~/Scripts/bootstrap.min.js",
"~/Scripts/bootbox.min.js",
"~/Scripts/jquery.tinyscrollbar.js" ) );
If you're developing a block, and you need to register a JavaScript on the page in a way that will ensure that it does not get added more than once, there are a couple different ways to do it. We've exposed a method call on the CurrentPage
object that will allow you to register your external JS file without worrying about collisions.
protected void OnLoad( EventArgs e )
{
base.OnLoad( e );
this.CurrentPage.AddScriptLink( this.Page, "~/Scripts/path-to-your-file.js" );
}
Under the covers, this method uses ASP.NET's built in ScriptManager
control to handle registering all JavaScript with the current page, and will ensure the same script file does not get added to the DOM more than once.
We try to adhere pretty closely to Doug Crockford's JS style guide. We use var
in front of all of our declarations, manually hoist our variables, and try to never use eval
, etc.
Brace placement, casing, etc
In our JavaScript code, the Rock team uses the K&R brace placement style. We use camelCasing for all variable declarations, and PascalCasing for "class" or module declarations. That style is pretty standard throughout the JavaScript community, and we believe that you shouldn't be able to tell what a developer's favorite server stack is by looking at their JavaScript.
Manual variable hoisting
The current version of JavaScript supported in modern browsers does not have block-level scope. So variables declared within code blocks, like for
loops or if
blocks are visible outside of that scope.
if (true) {
var a = 3;
}
// This will actually output '3' to the console.
console.log(a);
JavaScript has lexical (or function) scope. A variable will be available within the scope of the function it was declared, and any child functions declared within that scope.
To avoid any confusion it's considered good practice to manually hoist all of your variable declarations to the top of the function in which it was declared.
function foo () {
var a = 3,
b = function () {
var c = 0,
i;
for (i = 0; i < 5; i++) {
c += i;
}
return c;
};
console.log(a + b());
}
Equality checks and type coercion
When checking for equality of values, it's best to use the "triple equals" notation, rather than "double equals". When using "double equals" on values of two disparate types, JavaScript will attempt to cast the second operand to the first one's type in order to determine whether that statement is "truthy" or "falsy".
// Do this...
if (a === b) {
// ...
}
// And this...
if (a !== b) {
// ...
}
// Not this...
if (a == b) {
// ...
}
// Or this...
if (a != b) {
// ...
}
The reasoning is that "triple equals" will not attempt to use type coercion to evaluate the values being compared. If you're comparing two disparate types, they will remain two disparate types.
Here are a few examples of why "double equals" can be dangerous:
'' == 0
> true
'' === 0
> false
0 == false
> true
0 === false
> false
'1' == 1
> true
'1' === 1
> false
undefined == null
> true
undefined === null
> false
When checking the "truthiness" of a value, and you don't know what type it might be (e.g. - it could be an object, or a string), it's best to test the "truthiness" of the variable itself, rather than comparing it to multiple values:
// Do this...
if (foo) {
//...
}
// And this...
var a = foo ? 1 : 0;
// Not this...
if (foo !== null && (typeof foo === 'string' || typeof foo === 'object')) {
// ...
}
// Or this...
var a = (foo !== null && (typeof foo === 'string' || typeof foo === 'object')) ? 1 : 0;
When testing the existence of a member or an object, it's considered best practice to check for its type, rather than checking whether or not it's null:
// Consider this case...
var foo = {
bar: function () {
console.log('hi there!');
}
};
if (someCondition) {
foo.bar = 'baz';
}
// Do this...
if (typeof foo.bar === 'function') {
// This is very durable. It will never throw an error.
foo.bar();
}
// Not this...
if (foo.bar !== null) {
// If foo.bar === 'baz', the next line will throw an error.
foo.bar();
}
You will also run into logical operators being used in expressions. They can be used to evaluate the truthiness of the operand. You'll see the or ||
operator used in similar fashion to C#'s null coalescing operator ??
.
// If window.Rock is not `undefined`, `null`, `false`, `0` or `''`, window.Rock will be assigned.
// Otherwise, Rock will be assigned the value of a new object.
var Rock = window.Rock || {};
Strict mode
It's considered good practice to put your development code into strict mode. To do this, include the 'use strict';
pragma at the top of your outermost function closure in all of your JS files. This will put constraints on what the browser will allow you to do. The browser will now perform handy "compile-time" error checking on your code, which includes things like throwing an error if you forget a var
keyword in front of a variable declaration or define duplicate properties on an object.
Within Rock's client API's we make heavy use of the Module Pattern. You can think of a JavaScript "module" as being similar to a Singleton. However, since JavaScript in the browser is currently single-threaded, there's no need to worry as much about concurrency, locking, etc. Here's an example of the anatomy of a JavaScript module as we use it in Rock:
// We wrap our entire code block in an anonymous, self-calling function, or "closure".
// You will often see global variables passed in to ensure their durability, and encapsulate
// them within the closure. It's important to remember that all code defined within the
// closure is inaccessible from outside. This helps keep each module neatly encapsulated.
(function ($) {
// Make sure to put your code into strict mode...
'use strict';
// Establishing proper namespacing. This helps minimize collisions in the global scope.
window.Rock = window.Rock || {};
Rock.someNamespace = Rock.someNamespace || {};
// And here's our module. Notice it's another self-calling anonymous function.
// Whatever our anonymous function returns will be accessible from the global
// scope like so: `var foos = Rock.someNamespace.someModule.findFoos(5);`
Rock.someNamespace.someModule = (function () {
// This is a private function. It is not accessible from outside the module,
// but can be freely accessed from inside.
var _findFoos = function (num) {
var foos = [],
i;
for (i = 0; i < num; i++) {
foos.push('foo_' + (i + 1));
}
return foos;
},
// Here's our return value, enabling our module to have an external interface.
// Naming our return value exports is a convention we've taken from the Common.js
// module standard.
exports = {
findFoos: function (num) {
return _findFoos(num);
}
};
return exports;
}());
}(jQuery));
JavaScript: The Good Parts