Skip to content
This repository has been archived by the owner on Nov 3, 2019. It is now read-only.

Latest commit

 

History

History
897 lines (683 loc) · 26.8 KB

README.md

File metadata and controls

897 lines (683 loc) · 26.8 KB

MTS Code Guidelines

Version 2.0.0

I honestly didn't think you could even USE emoji in variable names. Or that there were so many different crying ones.

About these guidelines

The central goal of this document is to create readable, consistent code across web applications in MSS. While each application has its own constraints and needs, it is helpful to think of your application as not only a business-critical application, but also a tool for learning for all future developers!

Table of contents

JavaScript

The JavaScript guidelines are based off of idiomatic.js.

For lots of code examples that show the style we want, see the airbnb guide.

The MSS Guidelines will be the same with the following additional rules applied.

We will be using:

  • Mocha / Chai for unit tests
  • Selenium (WebdriverJS) / Mocha / Chai for functional tests
  • PhantomJS / CasperJS for functional tests

Idiomatic Style Manifesto

Whitespace

4-space soft indents are required. This means four spaces or four spaces representing a tab.

Place 1 space before the leading brace and 1 space before the parentheses in control statements (if, else, while, for)

if (things) {
    alert('Look how nice that space is!');
}

Place no spaces before the arguments list in a function declaration.

function makeSandwich(ham, cheese, egg) {
    return ham + cheese + egg;
}

Set off operators with spaces

var value = a + b + c;

var thirteen = (39 * 2) / 6;

User indentation when making long method chains (a la underscore/lodash, d3.js, etc.). Use a leading dot, which emphasizes that the line is a method call, not a new statement.

// bad
var leds = stage.selectAll('.led').data(data).enter().append('svg:svg').class('led', true)
        .attr('width', (radius + margin) * 2).append('svg:g')
        .attr('transform', 'translate(' + (radius + margin) + ',' + (radius + margin) + ')')
        .call(tron.led);

// good
var leds = stage.selectAll('.led')
                .data(data)
                .enter()
                .append('svg:svg')
                .classed('led', true)
                .attr('width', (radius + margin) * 2)
                .append('svg:g')
                .attr('transform', 'translate(' + (radius + margin) + ',' + (radius + margin) + ')')
                .call(tron.led);

Leave a blank line after blocks and before the next statement.

// bad
if (foo) {
    return bar;
}
return baz;

// good
if (foo) {
    return bar;
}

return baz;

// bad
var obj = {
    foo() {
    },
    bar() {
    },
};
return obj;

// good
var obj = {
    foo() {
       },

    bar() {
    },
};

return obj;

Declarations

Use only one variable declaration, at the top of your function. Chain together with commas

Why? Your functions should be short, and keeping the vars in one place is a good way to keep inventory as you read the function. Additionally, JavaScript will hoist all variable declarations in a scoped block to the top of the block anyway. For more information on hoisting, check out AirBnB's awesome writeup.

function doTwoThings() {
    var thingOne = 1,
        thingTwo = true;

    thingOne++;

    if (thingTwo) {
        thingOne++;
    }
}

Use single quotes for strings!

var myNameIs = 'Slim Shady';

Object literals should look like this:

var objectLiteral = {
    foo: 'bar',
    baz: 'qux'
};

Milliseconds should be assigned in multiples of 1000, and always use explicit order of operations.

var oneSecond = 1000 * 1,
    oneMinute = 1000 * 60,
    fiveMinutes = (1000 * 60) * 5;

Use a leading underscore _ when naming private properties.

// BAD! NO!
this._things_
this.things_
this.THINGS

// GOOD!
this._things = 'private';

Naming

Use camelCase for variable and function names

var thisVariableIsCamelCase = true;

function doSomethingGreat(isGreat) {
    if (isGreat) {
        alert('Pretty good!');
    }
}

Use PascalCase for constructor functions and classes.

function DoSomethingGreat(isGreat) {
    if (isGreat) {
        alert('Pretty good!');
    }
}

var greatThings = new DoSomethingGreat(true);

Be descriptive with your function and variable names!

// BAD! WHYYYY!!!
var nState = nStateOnLoad();

// GOOD!
var navigationState = getNavigationState();

Prefix jQuery selection variables with '$'

// BAD!
var navigationLinks = $('nav a');

// GOOD!
var $navigationLinks = $('nav a');

Comments

Use JSDoc-style comments to describe methods and functionality

/**
 * make() returns a new element
 * based on the passed in tag
 *
 * @param {String} tag
 * @return {Element} element
 */
function make(tag) {
    // some code
    return element;
}

Please use single line comments to describe how your code works! Debugging issues can be very difficult if it's unclear how a block of code works or what certain conditionals are for. Be kind to future developers!

function make(template) {
    var $element,
        isFancy;

    // first, create a jQuery instance of the html template string passed in
    $element = $(template);

    if (isFancy) {
        // if the element is supposed to be fancy, add a class
        $element.addClass('fancy');
    }

    return element;
}

Functions

The general approach when writing a function is that any future developer (including yourself in six months) can read it like a human and know what is happening. Comments should not be your crutch for sense-making- a combination of clear naming conventions, conditionals that read like sentences, and simple explanations in comments where needed will help keep code clear and maintainable. Example:

function makeSandwich(bread, meat, veggies) {
    // check if the sandwich has veggies, has meat, and determines sandwich potential
    var hasVeggies = Array.isArray(veggies) && veggies.length > 0,
        hasMeat = Array.isArray(meat) && meat.length > 0,
        weCanMakeASandwich = (bread && (hasVeggies || hasMeat)) ? true : false,
        message = 'No dude you can\'t make a sandwich';

    // sets defaults for meat and veggies
    meat = hasMeat ? meat : [];
    veggies = hasVeggies ? veggies : [];

    // if we can make a sandwich, tell us what it is all about
    if (weCanMakeASandwich) {
        message = 'Mmmm about to eat a ' + bread + ', ' + meat.join(', ') + veggies.join(', ') + ' sandwich';
    }

    return message;    
}
We like this function for several reasons:
  • The variable declarations contain all of the conditions we need to guarantee success.

  • All conditionals are consolidated, in an intentional order.

  • By using Booleans as our primary success criteria, we can write very readable execution code later on.

  • Tweaking the names of the variables lets us have human-readable conditionals ("if we can make a sandwich").

    • Originally, we had named this variable "canWeMakeASandwich", which, while readable, created an awkward sentence out loud.
  • A single return statement helps us reason about what this function outputs and when.

  • Always think about what you return- in a lot of cases, we can use this for chaining purposes.

  • And now, a very very bad example:

    function sandwich(ingredient1, ingredient2, veggies) {
        var message = '';
    
        // check if there is bread, or if there is only bread
        if (typeof ingredient1 === 'undefined') {
            message = "You can't make a sandwich without bread!";
            return message;
        } else if (typeof ingredient2 === 'undefined' && typeof veggies === 'undefined') {
            message = "You can't make a sandwich with only bread!";
            return message;
        }
    
        // if ingredient 2 exists but isn't an array, set it to empty
        if (typeof ingredient2 === 'undefined' || !Array.isArray(ingredient2)) {
            ingredient2 = [];
        }
    
        // if veggies exists but isn't an array, set it to empty
        if (typeof veggies === 'undefined' || !Array.isArray(veggies)) {
            veggies = [];
        }
    
        // if either ingredient 2 or veggies is an empty array
        if (ingredient2.length === 0 && veggies.length === 0) {
            message = "You don't have enough ingredients to make a sandwich, sir."
            return message;
        }
    
        // tell user what sandwich is being made
        message = 'Mmmm about to eat a ' + ingredient1;
    
        // if ingredient 2 exists, add to sandwich
        if (ingredient2.length > 0) {
            message = message + ' ' + ingredient2.join(', ');
        }
    
        // if there are veggies, add to sandwich
        if (veggies.length > 0) {
            message = message + ' ' + veggies.join(', ');
        }
    
        message = message + " sandwich.";
    
        // if the message is not an empty string, return a message
        return message == '' ? 'No dude you can\'t make a sandwich' : message;
    }
Some notes on this thing:
  • First, sorry!
  • The function name and arguments are not very descriptive. Is sandwich even a verb??? WTF is ingredient2???
  • All of the conditionals you see were 'conditional' on our understanding of the parameters needed for the function to succeed. tldr; We started with one, and quickly ballooned into this tangled web of logic.
  • While this function returns a slew of "useful" error messages, the basic question of "can i make a sandwich" does not require such verbose treatment. tldr; KEEP IT SIMPLE
  • The first fifteen some odd lines of conditionals in this function were accomplished in FOUR in the first function. This was done by creating readable, boolean variables to keep track of the essential information needed to run the function.
  • By using poorly named variables and doing explicit checks in a sequence of if/else conditionals, it is very easy to lost context for what is happening.
  • Assembling the message over multiple lines means that any future updates to the output of this function will most likely break the meaning.
  • Many return statements makes it difficult to tell when/if the function will stop executing and what will happen when it does.
  • Comments! The comments in this function do not add any value, mostly describing the exact code that follows. Sometimes not even that :)

HTML

Semantic Tags

Use semantic tags in the way they were intended!

HTML Tag Glossary

`section`
A meaningful division of content - every major group of content in between a header and footer
`header`
The header for a section
`footer`
The footer for a section
`article`
A full article
`nav`
Navigation! Main site navigation, contextual navigation - used for accessing site contents
`aside`
Elements that exist outside of the normal page flow/reading flow. Sidebars, related content, etc.
`cite`
A citation - used for references or bylines
`figure`
An image, illustration, or graph that is used as a meaningful piece of content
`figcaption`
The contextual caption for a ``

Example

Markup using semantic tags:

<!-- EXAMPLE MARKUP -->
<section class="recent-articles">
    <header>Most Recent Article</header>

    <nav>
        <a href="#allthearticles">Load all articles</a>
    </nav>

    <article>
        <h2>The Most Important Article</h2>

        <figure>
            <img src="/someimage.jpg" />

            <figcaption>Fig. 1 - This image explains everything!</figcaption>
        </figure>

        <cite>By Antonio Banderas</cite>

        <p>
            This is the article copy. It's pretty short because
            this is an example, and not for the internet at large.
            Remember, words are important!
        </p>
    </article>

    <footer>
        <a href="#signup">Sign up for our newsletter!</a>
    </footer>
</section>

Structural Elements

It is sometimes necessary to wrap HTML in tags that are structural, and have no direct impact on the content presented.

This is totally fine, but keep it clean!

In the example below, we have a UI that contains two buttons and a paragraph of text that is output based on some kind of user interaction. On a small screen (or small space), the buttons and the output text can stack vertically (think, display: block down the page).

On a larger screen, however, we may want these groups of elements to go 50/50 to better occupy screen real-estate.

In this type of case, we can semantically divide our html into two "ui-group"s, allowing us the flexibility required to make a nice app.

<section class="admin-tool">
    <div class="ui-group">
        <button>Button 1</button>
        <button>Button 2</button>
    </div>

    <div class="ui-group">
        <p>
            This is a read-out of some kind of
            admin tool. Clicking those buttons
            affects the text!
        </p>
        <a href="#reload">Reload That Text!</a>
    </div>
</section>

Data Attributes

Data attributes should be used to provide information to JavaScript to initialize or operate on that portion of the DOM. For instance, consider this fictional example that loads 20 comments from a particular service. At a high-level, all that is required to make the service calls are an id and a count for number of comments.

<div class="comments" data-comment-id="jks34" data-comment-count="20">
    ...
</div>

In special cases, data attributes can be used for CSS display values. Consider this example:

<div class="city--label" data-city-name="Houston, TX"></div>

<style>
    .city--label:before {
        content: attr(data-city-name);
    }
</style>

ID's

Use ID's sparingly! Make sure there is only one per document! And never style them ;)

Syntax

  • Always use semantic tags!

  • Always use doublequotes for attributes!

  • Always use proper indentation (keep your structure sane!)

  • Always make your HTML human-readable

  • Always put each new tag on its own line

  • Never use an #ID for styling hooks

  • Never use tables for layout. It's not 1999, this shouldn't be news. Tables are great for tabular data, though.

  • Never close <li> elements. <li> elements should not be closed. Further reading on this inline-block issue.

    <!-- BAD! NO! -->
    <div class="thing">
        <section><h1>Title</h1>
        <a href="thing">Things are happening <span>here</span></a></section>
    </div>
    
    <!-- Ah! Room to breathe, and now the structure is apparent. -->
    <div class="thing">
        <section>
            <h1>Title</h1>
    
            <a href="thing">
                Things are happening
                <span>here</span>
            </a>
        </section>
    </div>
  • Separate independent but loosely related snippets of markup with a single empty line, for example:

    EXCEPTION: For inline block styling, items may need to live on the same line. Acceptable deviance is inserting html comments to eliminate the inline-block space.

    <!-- THIS IS FINE, SINCE THESE TITLES AND PARAGRAPH ARE RELATED -->
    <div class="thing">
        <h1>Title</h1>
        <h2>Title 2</h2>
        <p>Things</p>
    </div>
    
    <!-- THIS IS ALSO FINE, GROUPS HEADLINES TOGETHER, AND SEPARATES THE PARAGRAPH -->
    <div class="thing">
        <h1>Title</h1>
        <h2>Title 2</h2>
    
        <p>Things</p>
    </div>
    
    <!-- THIS IS NOT FINE! DISTINGUISHES THESE ELEMENTS UNNECESSARILY -->
    <div class="thing">
    
        <h1>Title</h1>
    
        <h2>Title 2</h2>
    
        <p>Things</p>
    
    </div>
    
    <!-- SEPARATES LARGER "CHUNKS" OF HTML FOR EASE OF TRACKING -->
    <div class="thing">
        <nav>
            <a href="#">Link 1</a>
            <a href="#">Link 2</a>
        </nav>
    
        <h2>Title</h2>
    
        <figure>
            <img src="image.jpg"/>
            <figcaption>Caption for the image</figcaption>
        </figure>
    </div>

CSS/SASS

  1. The CSS/Sass guidelines are based off of csswizardry/CSS-Guidelines.
  2. We use normalize.css as our style reset.
  3. We favor BEM (Block Element Modifier) syntax where possible, and the wonderfully flat selector structure this gives us.
```css
/* Block */
.article {
    font-size: 1em;
}

/* Modifier */
.article--video {
    /* some video article styles */
}

/* Element */
.article__title {
    font-size: 3em;
}

/* Element */
.article__byline {
    color: #c00;
    font-size: .8em;
    font-weight: bold;
}
```

CSS Guidelines

Mobile First

Default styling must be mobile/small-sized first.

/* Poor */
.class {
    border: 1px solid #888;
    color: #ccc;

    @media (max-width: 359px) {
        border: none;
    }
}

/* Good */
.class {
    color: #ccc;

    @media (min-width: 360px) {
        border: 1px solid #888;
    }
}

Structure

Separate structural properties from visual properties using scoped mixins. This will allow us to easily swap visual appearance of elements without needing to override css definitions, and helps us think about skinning properties vs layout properties. This format also serves as a self-documenting table of contents.

For this to work, define a mixin within a class definition. Immediately after, include the mixin.

This may seem redundant, however, it helps us think about skinning properties vs layout properties.

// Color Vars
$robot-black: #121212;
$robot-gold: gold;
$robot-titanium: #ddd;

// Blocks
.robot {
    @mixin structure {
        display: block;
        height: 60px;
        width: 100px;
    }

    @mixin visual {
        background-color: $robot-titanium;
        border: 1px solid $robot-gold;
        box-shadow: 2px 2px 2px rgba(255,0,0,.6);
        transform: translateZ(1000px);
    }

    @include structure;
    @include visual;
}

// Elements
.robot__head {
    @mixin structure {
        display: block;
    }

    @mixin visual {
        background-color: $robot-black;
        border-radius: 1000px;
    }

    @include structure;
    @include visual;
}

.robot__claw {
    @mixin structure {
        display: block;

        &:before {
            content: 'CLAW TIME';
        }
    }

    @mixin visual {
        background-color: $robot-black;
        border-radius: 1000px;
        box-shadow: 10px 10px 10px gold;
    }

    @include structure;
    @include visual;
}

// Modifiers
.robot--space-robot {
    @mixin visual {
        background: url(/images/spacegraphic.png) repeat top left;
    }

    @include visual;
}

CSS Properties List

Here's a list of structural properties vs visual properties:

Structural Visual
display background
height box-shadow
margin border
padding color
position transform
width transition
z-index opacity
visibility

Depending on the application, here is a list of properties that could be used to define a theme:

Theme
background
border
box-shadow
color
font-family
text-transform
and more! (but nothing structural)

Markdown

  1. All lines that are not code blocks should wrap at or under 80 columns.
  2. Should allow trailing whitespace, since that is a valid Markdown syntax.
  3. Should have 2 blank lines above each H1 and H2 heading except for the heading at the top of the document. This is optinal for other headings.
  4. Should have 4 blank lines to separate the link reference at the bottom of the document.
  5. Should use 0. for ordered lists.
  6. Should use - for unordered lists.
  7. Secondary bullets must be indented four spaces to render correctly on Bitbucket.
  8. Do not try to put code blocks as secondary bullets in a list. They will not render correctly on both GitHub and BitBucket. More details on this to come in the future.

Contributors

YOU! - please contribute to these code guidelines! If you have any suggestions for improvement to these guidelines, create an issue to discuss it or create a pull request with your changes and discussion will occur on that PR. Also review the contributing guidelines.

Conflict Resolution

We have included [JSHint][jshint] and [JSCS][jscs] RC files in this repository to validate javascript code against these guidelines. When something is questioned, these files will always win. When in doubt, run JSHint and JSCS with the .jshintrc and .jscsrc files in place and see if they find any problems. It is recommended that you find plugins for your editor of choice to always validate your code against these files.

Regarding the multiple suffixed files

There are several .jscs and .jshintrc files with suffixes in this repo. It is up to you to choose the most appropriate one for your project. Hopefully the suffixes are clear, but just in case, here are some more details.

  • .jscs: A JavaScript Code Style config file for ECMAScript 5 code.

  • .jscs-es6: A JavaScript Code Style config file for ECMAScript 6 code.

  • .jshintrc-browser: A JS Hint config file for browser based ECMAScript 5 code.

  • .jshintrc-node: A JS Hint config file for NodeJS 0.10.x ECMAScript 5 code.

  • .jshintrc-node-es6: A JS Hint config file for NodeJS 0.11.x+ and IO.js ECMAScript 6 code.

Logging Levels

This is intended to be a short developer note on guidance for logging levels. Exception handlers should log exceptions.

Background

A brief synopsis of why we log and the implications for logging levels.

  • Hunt down bugs

  • Enable automated monitoring for errors and warnings and/or trigger Hipchat alerts

  • Maintain an audit trail

  • Record application status

  • Debugging configuration issues

IMPORTANT

  • Used sparingly

  • Used on production

  • Example: System start/restart

FATAL

  • Used for recording a failure that prevents system from starting or shuts down an application

  • Used on production

  • Triggers a DOC alert

  • Example: Systems becomes unusable

ERROR

  • Records when failure occurs but system is stll usable

  • Used on production

  • Triggers a DOC alert

  • Example: PAL errors talking to IBIS

WARNING

  • A scenario that indicates something is wrong with the system but does not prevent system from operating correctly

  • Used on production

  • Monitored and triggers DOC if defined levels exceeded

  • Example: Invalid login attempts

INFO

  • General system status— Does not indicate something is wrong

  • Used on production

  • Should be able to be enabled in production

  • Example: Reestablish connection to external service

DEBUG

  • Enables developer context (stack trace and/or other context of issue) for debugging a problem

  • Should not be used in production for performance as well as security reasons.

  • Example: Null pointer exception

VERBOSE

  • Should be used for capturing detailed implementation and diagnostic level detail that you might want to enable in rare situations (a more verbose INFO)

  • Not used in production

  • Touch base with your Tech Lead for this level of logging

SILLY

  • Should be used for capturing very detailed implementation and diagnostic level detail that you might want to enable in very rare situations (a more verbose DEBUG)

  • Not used in production

  • Touch base with your Tech Lead for this level of logging

Regarding the multiple suffixed files

There are several .jscs and .jshintrc files with suffixes in this repo. It is up to you to choose the most appropriate one for your project. Hopefully the suffixes are clear, but just in case, here are some more details.

  • .jscs: A JavaScript Code Style config file for ECMAScript 5 code.

  • .jscs-es6: A JavaScript Code Style config file for ECMAScript 6 code.

  • .jshintrc-browser: A JS Hint config file for browser based ECMAScript 5 code.

  • .jshintrc-node: A JS Hint config file for NodeJS 0.10.x ECMAScript 5 code.

  • .jshintrc-node-es6: A JS Hint config file for NodeJS 0.11.x+ and IO.js ECMAScript 6 code.

Helpful Links

AirBnB Style Guide
CSS Guidelines
Idiomatic Javascript
Inline Block Problems!
GitHub Issues
JSCS
JSHint
Normalize CSS
Pull Requests