Type safe Angular with TypeScript
Nick Tsitlakidis
Software Developer @ AMD Telecom, Routee team
Agenda
- What is TypeScript
- Ok, but I work on Java APIs
- Existing libraries
- Mixing with Angular
- Tooling
Yet another *Script?
JavaScript with types and a helpful compiler
What is it?
- Open source with all development happening on Github.
- Introduced by Microsoft but more are actively contributing.
- JavaScript superset
- Valid JavaScript is valid TypeScript too.
- Focusing on ES features to be available in the future.
- Adds classes, interfaces, (proper) inheritance etc, but...
- They are optional!
Basic Types
- boolean
- number
- string
- array
- tuple
- enum
- any
- void
Syntax
var counter:number = 0;
var counterStarted:boolean = (counter != 0);
var counters:Array<string> = [];
counters.push(counter.toString());
counter will be treated as numeric
Superset of JavaScript
The good!
- You can migrate your code at your own pace
- Compiler output is readable (and debuggable)
- Most of the features will be available in future versions of JavaScript.
- No special runtime required
- Compiles to ES3 / ES5 / ES6 (ES2015)
Superset of JavaScript
The complicated!
- Your build process needs new steps (compilation)
- You need to decide about module systems
- You need to learn about source maps if you want to debug easily
- Some OOP concepts are not what you're used to
Superset of JavaScript
The code
function addTwoNumbers(one,two) {
return one+two;
}
var result = addTwoNumbers(3,6); //this will work
var anotherResult = addTwoNumbers("3",6); //returns "36"
function addTwoNumbers(one:number,two:number):number {
return one+two;
}
var result:number = addTwoNumbers(3,6); //this will work
var anotherResult:number = addTwoNumbers("3",6); //your IDE shows an error at compile time
OOP features
The good!
- ES2015 Classes
- Interfaces
- Abstract classes
- Polymorphism
- Generics
- Access modifiers
- Decorators
- No method overloading
- No constructor overloading
- Interfaces are gone at runtime
- Types are gone at runtime
The complicated!
OOP features
The code!
interface CoffeeConsumer {
drinkCoffee();
}
class Developer implements CoffeeConsumer {
private name:string;
constructor(name:string) {
this.name = name;
}
public drinkCoffee() {
this.takeABreak();
this.prepareCoffee();
}
private takeABreak() {
console.log("hurray, a break");
}
private prepareCoffee() {
console.log("where is the sugar?");
}
}
var exhaustedDev:CoffeeConsumer = new Developer("Dev");
exhaustedDev.drinkCoffee();
var Developer = (function () {
function Developer(name) {
this.name = name;
}
Developer.prototype.drinkCoffee = function () {
this.takeABreak();
this.prepareCoffee();
};
Developer.prototype.takeABreak = function () {
console.log("hurray, a break");
};
Developer.prototype.prepareCoffee = function () {
console.log("where is the sugar?");
};
return Developer;
}());
var exhaustedDev = new Developer("Dev");
exhaustedDev.drinkCoffee();
Different interface usages
interface Person {
name:string;
age:number;
}
var p:Person = {
age:30,
name:"Nick"
};
interface Person {
name:string;
age:number;
}
class User implements Person {
public name:string;
public age:number;
}
var p:Person = new User();
p.age = 30;
p.name = "Nick";
function addPerson(person:{ name:string; age:number; }) {
console.log(person.age);
}
Direct object creation
Inline interface usage
Implemented by a class
Why it matters for APIs
@Getter
@Setter
public class ContactDto {
private String id;
private String firstName;
private String lastName;
}
Your API response
$.getJSON("/contacts/345", function(data) {
//do something with the response data
});
Your AJAX call
interface ContactDto {
id:string;
firstName:string;
lastName:string;
}
$.getJSON( "/contacts/345", (data:ContactDto) => {
console.log(data.firstName);
console.log(data.lastName);
console.log(data.id);
});
The problem with JS libraries
- They are countless so we need to take advantage of them
- We can't rewrite them
- Writing adapters is also hard depending on the language
- Ambient declarations to the rescue!
- Declare syntax
Ambient Declarations
interface LoDashStatic {
forEach<T>(
collection: T[],
iteratee?: ListIterator<T, any>,
thisArg?: any
): T[];
groupBy<T, TKey>(
collection: List<T>,
iteratee?: ListIterator<T, TKey>,
thisArg?: any
): Dictionary<T[]>;
...
}
declare var _:LoDashStatic
interface Angular {
controller(name:string, controllerClass:Function);
}
declare var angular: Angular;
angular.controller("MyController", MyControllerClass);
Typings
npm install -g typings
- Typings manager available through npm
- Typings for both the browser and node.js
- Store the typings you need in a definitions file
- Included through reference paths or tsconfig.json
typings install lodash --save
///<reference path="typings/browser.d.ts"/>
_.forEach([1,2,3],(value)=> {
console.log(value);
});
No compiler errors + code hinting
Type safe Angular.js
- Services can be classes
- Controllers can be classes (using "controller as" syntax)
- Lambdas can be used to avoid issues with "this"
- Decorators can be used for common class tasks (injections)
- Your models can now be described as interfaces or classes
Angular controller classes
interface User {
firstName;
}
class UserProfileController {
public user:User;
private httpService:ng.IHttpService;
constructor($http:ng.IHttpService) {
this.httpService = $http;
this.httpService({method: 'GET', url: 'http://my-api.com/users/356'})
.then((user:User)=> {
this.user = user;
});
}
}
angular.module("Profile").controller("UserProfileController",UserProfileController);
No more $scope! (most of the times...)
<div ng-controller="UserProfileController as controller">
<p>{{controller.user.firstName}}</p>
</div>
Angular service classes
class UserService {
private httpService:ng.IHttpService;
constructor($http: ng.IHttpService) {
this.httpService = $http;
}
public getUser(userId:string):ng.IPromise<User> {
return this.httpService({
method: 'GET',
url: 'http://my-api.com/users/'+userId
});
}
}
angular.module("Services").service("UserService",UserService);
Angular services with Restangular
class UserService {
private restangular:restangular.IService;
constructor(Restangular:restangular.IService) {
this.restangular = Restangular;
}
public getUser(userId:string):ng.IPromise<User> {
return this.restangular.one('users', userId).get();
}
public updateUser(user:User):ng.IPromise<User> {
return this.restangular.one("users", user.id).customPUT(user);
}
}
- Centralized configuration for all your api calls
- Interceptors for responses and requests
- Centralized authorization flow
- Centralized error handling
- Methods are based on RESTful concepts
Tools
- Editors and IDEs
- Jetbrains (Intellij IDEA / Webstorm/ PHPstorm)
- Microsoft (Visual Studio / Visual Studio Code)
- Atom
- Sublime Text
- NetBeans
- Eclipse
- TypeScript Playground
- Webpack
- Browserify
- TypeDoc
- Grunt / Gulp
Type safe Angular using TypeScript
By Nick Tsitlakidis
Type safe Angular using TypeScript
A little bit of Angular.js, a lot about TypeScript, a hint at what we do and a look into the future.
- 2,155