The intention of this document is to standardise JavaScript and CSS naming conventions and syntaxes across our frontend repositories. Don’t shoot the messenger. Many subheadings also includes a “Why?” section to explain our rationale.
Variable names should be self-descriptive nouns – reflecting the value of the variable and negating the need for comments describing the value.
// don't do this
const value = 2020; // current year
// do this instead
const currentYear = 2020;
If a variable’s value will not change, define it with const
to make it immutable. Otherwise, unless necessary, use let
in order to declare variables in the smallest scope possible. There are very few cases where you will need to use var
.
Try to keep variable names short, but don’t sacrifice descriptiveness in fear of verbosity. With this in mind, avoid abbreviating variable names.
// don't do this
const bioDesc = "I hate puppies";
// do this instead
const biographyDescription = "I love puppies";
Variable names should be written in camelCase.
// don't do this
const first_name = "Luke";
// do this instead
const firstName = "Lucrecio";
The same goes for object properties and methods.
// do this
const newUser = {
firstName: "Albert",
lastName: "Camus",
};
On occasion, we’ll need to handle snake_case properties in API responses, but they should be converted to camelCase before leaving their associated service
.
Why?
In general, the goals with a variable declaration are to keep code clear, concise and easy to read. Historically, JavaScript developers sought short variable names because screens were narrow and minification was not a widespread practice. Neither of these concerns exists today and, instead, we want every developer to have as much information as possible with as little effort as required – allowing for quick onboarding, readable interfaces and efficient debugging.
The primary exception to the above camelCase rule is when defining constants for use throughout a component, page or the whole application. These should be defined in all caps with underscores used to separate words, what is known as Screaming Snake Case or, as I call it, My Uncle On Facebook Case.
// do this
const ACCOUNT_ACTIVATION_ENDPOINT = "/acccount-activation";
Why?
Just as with variables more generally, the intent with using screaming snake case is to make constant definitions as clear as possible to other developers as early as possible so they can get to solving problems and closing tickets as fast as possible.
In most cases – all, unless it forces you to negate the principle of self-descriptiveness – boolean definitions should begin with is
, has
, can
or are
in order to signal to other developers that the value is boolean.
// don't do this
const redBackground = true;
// do this instead
const isRedBackground = true;
Also, we should avoid passing booleans as function parameters without context.
// don't do this
function getColor(isRed: boolean) {
return isRed ? "#913122" : "#2b6b46";
}
const color = getColor(true);
// do this instead
function getColor(color: "red" | "green") {
return color === "red" ? "#913122" : "#2b6b46";
}
const color = getColor("red");
// or this
function getColor({ isRed }: { isRed: boolean }) {
return isRed ? "#913122" : "#2b6b46";
}
const color = getColor({ isRed: true });
Why?
Our goal is to make it clear as quickly as possible that a boolean variable is, in fact, a boolean and remove the need for our coworkers to go hunting for value declarations.
Like variables, function names should be self-descriptive. Unlike variables, they should be verbs instead of nouns. This means they should describe the action the function will undertake.
// don't do this
function hungover(tipo: Human, drunk: number) {
while (tipo.cuerpito < drunk) {
drinkBeer(tipo);
}
return sleepUntilMorning(tipo);
}
// do this instead
function getHungover(tipo: Human, drunk: number) {
...
}
// don't do this
const removeTrailingCommas = (name: string) => name.replace(/,\s*$/, "");
// do this instead
function removeTrailingCommas(name: string) {
return name.replace(/,\s*$/, "");
}
// don't do this
export default function (name: string) {
return `Hello ${name}!`;
}
// don't do this either... even though it looks cool
export default (name: string) => `Hello ${name}!`;
// do this instead
export default function getGreeting(name: string) {
return `Hello ${name}!`;
}
Not only will this make your functions easier to read, but it will also help with debugging errors in your stack traces.
A notable exception is lambda functions, where anonymous functions are excellently suited for callbacks.
// do this
interface Student {
name: string;
age: number;
}
const studentAges = students.map(({ age }: Student): Array<number> => age);
Why?
As with our other rules, the main goal here is clarity. However, another benefit to using function declarations is hoisting, which allows you to use your function before you declare it.
// don't do this
function greet(name, surname, age, nationality) {
return `Hello ${name} ${surname}, you're ${age} years old and from ${nationality}`;
}
// do this
function greet(args) {
const { name, surname, age, nationality } = args;
return `Hello ${name} ${surname}, you're ${age} years old and from ${nationality}`;
}
We encourage you to name the object args for clarity and destructure it on a single to make the function easier to read.
Why?
- Clarity: When calling a function with multiple parameters, it’s helpful to know what the parameters are about.
- Less error prone when not typescript.
- The order of the parameters is not important anymore.
- It’s easier to add new parameters.
- It’s easier to skip optional parameters.
greet("Lisa", "Simpson", 12, "Springfield");
For instance, in this case, is not obvious what 12 refers to. It’s a number but it refers to an age, a day of the month, an amount? Using an object, we emulate named parameters, hence every option the function operates upon on gets “labelled”:
greet({ name: "Lisa", surname: "Simpson", age: 12, city: "Springfield" });
It allows us to understand better the function without having to look at its implementation details.
We can as well change the parameter order without side effects or skip some of them without the function invocation looking weird.
JavaScript provides no native way to declare a private variable or method – though there are multiple old school and new school approaches to solving that problem. We define private fields and functions as any function that will not be exported by a component or module’s interface and we use the common convention of prefacing their name with an underscore.
// do this
function _getIsHidden() {
return state.errorMessage === '';
}
export {
isHidden: _getIsHidden(),
}
Why?
Again, the goal here is to achieve clarity and efficiency. Prefacing private fields with an underscore signals to other developers that any changes made that field/function will only affect a reduced scope (eg. the current module/class), which makes it easier to debug, refactor and avoid side effects.
As opposed to classes, services are files that export an object containing non-instantiable, usually pure functions. Services can be thought of as collections of utilities grouped around completing a single, often generic task. We aim to keep service functions pure and free of side effects.
// do this
function getHasLowercaseChar(password: string) {
return /[a-z]/.test(password);
}
function getHasMinimumLength(password: string) {
return password.length >= MIN_REQUIRED_LENGTH;
}
function getHasNumber(password: string) {
return /\d/.test(password);
}
function getIsValidPassword(password: string) {
const hasLowercaseChar = getHasLowercaseChar(password);
const hasMinimunLength = getHasMinimumLength(password);
const hasNumber = getHasNumber(password);
return hasLowercaseChar && hasMinimunLength && hasNumber;
}
export default {
getHasLowercaseChar,
getHasMinimumLength,
getHasNumber,
getIsValidPassword,
};
The naming convention for a service follows the pattern ***.service.ts
, where ***
is a self-descriptive name describing the purpose of the service. In the above example, the name might be password-validation.service.ts
.
Also, services should have a default export containing every exported function.
Why?
Separating out service logic from components server the dual purpose of extracting logical, stateless concerns from the UI and possible side effects, while also making it easy to set up a test double on a default export.
// don't do this
interface Card {
provider: "VISA" | "MASTERCARD";
pan: number;
}
// do this
interface ICard {
provider: "VISA" | "MASTERCARD";
pan: number;
}
This is a polemic topic. The official typescript documentation discourages the use of I prefixes however in my experience, it is a good practice to use them because under some circumstances it can lead to unexpected bugs.
Once I was working on a project where the Card class existed and represented a React component Card
but the auto-importer imported the Card
interface instead of the Card
component. This caused a lot of confusion and wasted time.
<Card>
<CardHeader />
<CardBody />
<CardFooter />
</Card>
While variables are named using camelCase, files should be named using a snake_case or kebab-case conventions.
This is because sometimes git can play some tricks on you on case-insensitive filesystems.
https://stackoverflow.com/questions/10523849/changing-capitalization-of-filenames-in-git
https://www.reddit.com/r/git/comments/dj5b8n/lpt_check_your_uppercaselowercase_file_names/
Use an ending to indicate the content type of the file:
component.view.tsx
component.controller.tsx
component.spec.tsx
component.service.ts