ngular best practices
Wilson Mendes
@willmendesneto
Google Developer Expert Web technologies
data:image/s3,"s3://crabby-images/c0485/c0485df086779c13c99f22a7cfa9f26a1d20ac71" alt=""
Hi, it's me
DURING THE FREE TIME...
1. Access https://goo.gl/PibUcQ
2. Decrease the price to $0,00
3. Enjoy
Nodebots book free!
data:image/s3,"s3://crabby-images/166fc/166fc0d4731186052ab6c6406ee64485b89a0f13" alt=""
So, let's start?
<webapps/> are evolving
and we need to be up-to-date
Share components is <3
data:image/s3,"s3://crabby-images/7d30a/7d30a403750a3121e21b8418488ad2d8dfcee05d" alt=""
data:image/s3,"s3://crabby-images/8bf97/8bf97dcebdb6e40ab8e0d87de730111c433e2ec1" alt=""
data:image/s3,"s3://crabby-images/240e3/240e31ffbc503dd179345775e8d6ff19cc82b35a" alt=""
data:image/s3,"s3://crabby-images/f2baf/f2baf10a4024f07a635b9e7f84dcecceb44a642b" alt=""
Layout container
<page1>
</page1>
<component1 />
<component2>
</component2>
<component3 />
4
TIPS
TIPS
Nice to meet you
ngular Styleguide
Tip #1
data:image/s3,"s3://crabby-images/f0d17/f0d176353599497e91fe591c967bddb82e5de6a9" alt=""
Types of Modules
CORE
FEATURE
CUSTOM
SHARED
Lazy loading
EVERYTHING
@NgModule({
providers: []
})
class MyModule {
static forRoot() {
return {
ngModule: MyModule,
providers: [ /* LIST OF YOUR PROVIDERS */]
}
}
}
import { NgModule } from '@angular/core';
import { Routes, RouterModule } from '@angular/router';
import { MyModule } from './my.module';
export const routes: Routes = [
{
path: 'crisis',
loadChildren: 'app/crisis/crisis.module#CrisisModule'
}
];
@NgModule({
imports: [
RouterModule.forRoot(routes),
MyModule.forRoot()
],
exports: [RouterModule, RouterModule]
})
export class AppRoutingModule {}
npm install @herodevs/lazy-af
...
@NgModule({
imports: [LazyModule],
})
export class AppModule {}
...
<div (mouseover)="load = true">
Hover to load TestModule
</div>
<lazy-af
*ngIf="load"
moduleName="src/app/test/test.module#TestModule"
>
</lazy-af>
nalize your bundle
Tip #2
webpack-bundle-analyzer
lways small
Tip #3
Integration in your module
Easy to maintain
Refactor across teams
Why small packages
When you use a component ...
Demo page, please
Plug-and-play
that is how components should be 😉
...
import {
NgxSkeletonLoaderModule
} from 'ngx-skeleton-loader';
...
@NgModule({
...
imports: [NgxSkeletonLoaderModule],
...
})
export class AppModule { }
<ngx-skeleton-loader
count="5"
[theme]="{
'border-radius': '5px',
height: '50px'
}"
>
</ngx-skeleton-loader>
Decoupling Business Logic
EX: Feature toggle
data:image/s3,"s3://crabby-images/254bd/254bd2655a2228f48f9ea165e63b2e307b334f6f" alt=""
data:image/s3,"s3://crabby-images/07a87/07a87260250c01157af60e86cb4726246f9cca47" alt=""
Feature Toggle Service
Reactor Feature toggle
data:image/s3,"s3://crabby-images/3db40/3db40f0d74276cec44e3b714af86a0ab99147660" alt=""
data:image/s3,"s3://crabby-images/bf415/bf415c53f27895f0a5d45fe39f3ac0fe5c137f57" alt=""
NGX Feature toggle
import { FeatureToggleModule } from 'ngx-feature-toggle';
...
@NgModule({
...
declarations: [
YourAppComponent
],
imports: [ FeatureToggleModule ],
bootstrap: [YourAppComponent]
...
})
export class YourAppComponent {
}
...
<feature-toggle [featureName]="'enableContent'">
<p>Enabled content</p>
<feature-toggle [featureName]="'disabledContent'">
<p>Disabled content</p>
<p>In that case this content should not be rendered.</p>
</feature-toggle>
<feature-toggle
[featureName]="'disabledContent'"
showWhenDisabled >
<p>Disabled content</p>
<p>But, it has `showWhenDisabled` attribute.</p>
<p>In that case this content should be rendered.</p>
</feature-toggle>
</feature-toggle>
data:image/s3,"s3://crabby-images/2868e/2868e4c43de4080f3ce95387edf34e43c166788a" alt=""
data:image/s3,"s3://crabby-images/8697f/8697fbe62313fcdc384fe60740c6071b9513525a" alt=""
ngDoCheck: 4.7ms
data:image/s3,"s3://crabby-images/7353a/7353afa1a2a65ceb574e522da24580c17e0e89b9" alt=""
ngZone: 2.3ms
lways improve DX
Tip #4
Have you ever have problems with
Documentation
Deployments
Maintainers
Communication
Migration docs
Dependencies
Monorepos
...can be a good solution
Organized
Tooling
Tasks
Traceability
Cross-project
Dependencies
Why monorepos
data:image/s3,"s3://crabby-images/1d587/1d587e25e214131f146cef35ff66234658cf2538" alt=""
data:image/s3,"s3://crabby-images/56367/563672c0e8c506df713cd851f48285118a9f18d5" alt=""
yarn affected:build --base=master
// Running affected only
// E2E Tests
yarn affected:test --base=master
yarn affected:e2e --base=master
// Unit Tests
// Build
Greenkeeper
Code Climate
Publishing a module
...can give you some hard time
ng generate my-awesome-lib
// Generating a library
ng build my-awesome-lib --prod
cd dist/my-awesome-lib
npm publish
// Publishing a library
np
--no-yarn
--contents=dist
// Publishing a library
data:image/s3,"s3://crabby-images/7d18f/7d18f19043390ab61bdb5b73ac0727ac5f39b391" alt=""
data:image/s3,"s3://crabby-images/d2f97/d2f9791553153509a9d9f8234548c8c85e7d97aa" alt=""
Blue Green dep
Canary release
Dark launch
Feature Flag Rollout
Release Strategy
MASTER
BRANCH
Landkid
{1}
{2}
{3}
{4}
{5}
{2}
{6}
{5}
{1}
}
MASTER
BRANCH
{1}
{2}
{3}
{4}
{5}
{2}
{6}
{5}
{1}
}
MASTER
BRANCH
MASTER
BRANCH
{1}
{1}
{2}
{3}
{4}
{5}
{6}
}
{2}
{5}
{1}
MASTER
BRANCH
MASTER
BRANCH
<ngx-recap/>
Angular styleguide
Analyse your bundle
Improve dev experience
Automate everything
Thank you
Wilson Mendes
@willmendesneto
Google Developer Expert Web technologies
data:image/s3,"s3://crabby-images/c0485/c0485df086779c13c99f22a7cfa9f26a1d20ac71" alt=""