Avas LogoAvas

Unveiling Hidden Gems Lesser-Known Facts About TypeScript

Typescript is awesome language. I have been using it for one year now and I my love towards the language is infinity. It's commonly known as the superset of Javascript, but it's much more than that. Typescript helps a lot during the day to day development work, also it helps a lot to avoid the unnecessary painful bugs.

Here, in this articles I am going to present some features ina Typescript that you might not know and that may come handy in your day to day life. Are you excited to go through the features?

@ts-expect-error

Let's assume a case where you have a case where the ts compiling is yelling at you that you are doing something wrong,

const multiply = (a: number, b: number) => a + b;
multiply("12", 13);

Here, TS compiler yells at you with the message Argument of type 'string' is not assignable to parameter of type

Let's say you cannot change the type of the first argument and temporarily want to surpress the error being shown by the TS compiler, normally we use @ts-ignore to surpress the error

const multiply = (a: number, b: number) => a * b;
// @ts-ignore
multiply("12", 13);

Now, after some time, you fixed the error and changed the first argument of the multiply function from '12' to 12

const multiply = (a: number, b: number) => a + b;
// @ts-ignore
multiply(12, 13);

but, you forgot to remove the @ts-ignore directive we used previously, so basically it will keep ignoring the next line forever unless you remove it which may cause bug in the future.

So in this case, we can make use of @ts-expect-error directive, which does the same job as the @ts-ignore directive but as soon as the error is fixed, the TS compiler yells at you

const multiply = (a: number, b: number) => a + b;
// @ts-expect-error
multiply(12, 13);

Unused '@ts-expect-error' directive.

This reminds you to remove the directive as soon as the error is fixed.

never type

You have a function that takes an error status code and always throws an error according to the status, the never type comes handy when you know a function is never going to reach it's end point.

The difference between never and void is, while void means atleast undefined or null is being returned, never means the function's end point is never reached.

function throwError(error: string): never {
  throw new Error(error);
}

Template Literal Types

Tempplate literal types are similar to string litreral types in javascript but are specific to types. Let's say you have a library that implements a popover and there is a type for positioning the popover

type popoverPositions =
  | "top"
  | "bottom"
  | "left"
  | "right"
  | "top-left"
  | "top-right"
  | "top-center"
  | "bottom-left"
  | "bottom-right"
  | "bottom-center";

It can be hectic to combine all the possible permutations and combinations for all the types

using template literal types, you can separate the directions and combine the types to get a new type that consists of all the possible combinations

type positions = "top" | "bottom" | "center";
type directions = "left" | "right" | "center";
type popoverPositions = positions | directions | `${positions}-${directions}`;

which will generate all the types as

type popoverPositions =
  | positions
  | directions
  | "top-left"
  | "top-right"
  | "bottom-left"
  | "bottom-right"
  | "center-left"
  | "center-right";

Null assertions

Null assertions basically tells your TS compiler that your value is nor null neither undefined. Let's say you have initialized the value as let myNumber:null | number = null; but later on you update the value of myNumber myNumber = 69;

Now, lets assume you have a function that accepts only numbers,

const add = (a: number, b: number) => {
  return a + b;
};
add(myNumber, 1);

The compiler yells at you saying Argument of type 'null' is not assignable to parameter of type 'number'.

So here, you can make use of null assertions width bang ! in the end of the variable to tell the compiler that the value we are sending is not null

const add = (a: number, b: number) => {
  return a + b;
};
add(myNumber!, 1);

The code above compiles successfully.

Merging Interfaces

Megring Interfaces is the type of declaration merging, when you have two interfeces with the same name, it merges and creates a single interface

interface Box {
  height: number;
  width: number;
}
interface Box {
  scale: number;
}
let box: Box = { height: 5, width: 6, scale: 10 };

So here, we can create two separate interfaces with the same name which in turn gets merged to a single one and can be used as per the example mentioned above.

I hope, you learnt something new from the list above.