Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/nigrosimone/ng-portal
Component property connection in Angular application wherever they are.
https://github.com/nigrosimone/ng-portal
angular angular2 decorators rxjs
Last synced: about 2 months ago
JSON representation
Component property connection in Angular application wherever they are.
- Host: GitHub
- URL: https://github.com/nigrosimone/ng-portal
- Owner: nigrosimone
- License: mit
- Created: 2021-03-20T12:27:17.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-01-06T20:03:31.000Z (about 1 year ago)
- Last Synced: 2024-08-17T05:14:11.713Z (5 months ago)
- Topics: angular, angular2, decorators, rxjs
- Language: TypeScript
- Homepage: https://www.npmjs.com/package/ng-portal
- Size: 3.03 MB
- Stars: 16
- Watchers: 4
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
- awesome-angular - ng-portal - Component property connection in Angular application. (Table of contents / Third Party Components)
- fucking-awesome-angular - ng-portal - Component property connection in Angular application. (Table of contents / Third Party Components)
README
# NgPortal [![Build Status](https://travis-ci.com/nigrosimone/ng-portal.svg?branch=main)](https://travis-ci.com/nigrosimone/ng-portal) [![Coverage Status](https://coveralls.io/repos/github/nigrosimone/ng-portal/badge.svg?branch=main)](https://coveralls.io/github/nigrosimone/ng-portal?branch=main) [![NPM version](https://img.shields.io/npm/v/ng-portal.svg)](https://www.npmjs.com/package/ng-portal)
Component property connection in Angular application wherever they are.
## Description
Sometime there is a need to send data beetween components. A common pattern in Angular is sharing data between a parent component and one or more child components by using the `@Input()` and `@Output()` directives.
This pattern works if component are in the same scope.
In this example an _output component_ set a property `value` imputed by _input component_, eg:```ts
import { Component } from '@angular/core';@Component({
selector: 'app-component',
template: `
`,
})
export class AppComponent {
value: string;
}
```But what happen if both component arent in the same scope? A common pattern in this case is propagate `@Input()` and `@Output()` throught the tree of parents/childs component or write a shared service for exhange the data.
`NgPortal` offer a dead simple solution, a new directive `@NgPortal()` that connect two property wherever they are.
In this example every property called `value` with `@NgPortal()` directive is connected and every changes is propagated wherever, eg.:
```ts
import { Component } from '@angular/core';
import { ngPortalInput, ngPortalOutput } from 'ng-portal';@Component({
selector: 'app-input',
template: ``,
})
export class InputComponent {
@ngPortalInput() value: string;
}@Component({
selector: 'app-output',
template: `{{ value | async }}`,
})
export class OutputComponent {
@ngPortalOutput() value: Observable;
}
```There are also mono-directional directive `@NgPortalInput()` and `@NgPortalOutput()` for more control.
See the [stackblitz demo](https://stackblitz.com/edit/demo-ng-portal?file=src%2Fapp%2Fapp.component.ts).
## Features
✅ Two way data binging
✅ Mono directional communication
✅ Async pipe support
✅ NgModel support## Get Started
*Step 1*: install `ng-portal`
```bash
npm i ng-portal
```*Step 2*: Import `NgPortalModule` into your app module, eg.:
```ts
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { AppComponent } from './app.component';import { NgPortalModule } from 'ng-portal';
@NgModule({
declarations: [AppComponent],
imports: [
BrowserModule,
NgPortalModule,
],
providers: [],
bootstrap: [AppComponent],
],
})
export class AppModule { }
```## API
Available property decorators (`options` is optional):
- `@ngPortal(options?: NgPortalDecoratorOptions)`: two way communication
- `@ngPortalInput(options?: NgPortalDecoratorOptions)`: only send changes
- `@ngPortalOutput(options?: NgPortalDecoratorOptions)`: only receive changesDecorator options interface:
```ts
export interface NgPortalDecoratorOptions {
key: string;
}
```From default `ngPortal` use property name as the key:
```ts
@ngPortal() value: string;
```is equivalent to:
```ts
@ngPortal({key: 'value'}) value: string;
```and is also equivalent to:
```ts
@ngPortal({key: 'value'}) whateverYouWant: string;
```Communication ways:
- `@ngPortal()` sends to `@ngPortal()` and `@ngPortalOutput()`
- `@ngPortal()` receives from `@ngPortal()` and `@ngPortalInput()`
- `@ngPortalInput()` sends to `@ngPortal()` and `@ngPortalOutput()`
- `@ngPortalInput()` doesn't receive
- `@ngPortalOutput()` doesn't send
- `@ngPortalOutput()` receives from `@ngPortal()` and `@ngPortalInput()`Type:
Behind the scene, `@ngPortal()` apply a getter and a setter on the property:
- `@ngPortal() property: string;` => `get property(): Observable` and `set property(value: string): void`
- `@ngPortalInput() property: string;` => `set property(value: string): void` (only setter available)
- `@ngPortalOutput() property: string;` => `get property(): Observable` (only getter available)## Examples
Below there are some examples of use case.
### Example: input / output components connected by property name
`InputComponent` has property `value` with `@ngPortalInput()` decorator that on change update property `value` into `OutputComponent` with `@ngPortalOutput()`, eg.:
```ts
import { Component } from '@angular/core';
import { ngPortalInput, ngPortalOutput } from 'ng-portal';
import { Observable } from 'rxjs';@Component({
selector: 'app-input',
template: ``,
})
export class InputComponent {
@ngPortalInput() value: string;
}@Component({
selector: 'app-output',
template: `{{ value | async }}`,
})
export class OutputComponent {
@ngPortalOutput() value: Observable;
}
```### Example: input / output components connected by key
`InputComponent` has property `inputValue` with `@ngPortalInput({key: 'foo'})` decorator that on change update property `outputValue` into `OutputComponent` with `@ngPortalOutput({key: 'foo'})`. In this case is the key 'foo' that made the connection, eg.:
```ts
import { Component } from '@angular/core';
import { ngPortalInput, ngPortalOutput } from 'ng-portal';
import { Observable } from 'rxjs';@Component({
selector: 'app-input',
template: ``,
})
export class InputComponent {
@ngPortalInput({key: 'foo'}) inputValue: string;
}@Component({
selector: 'app-output',
template: `{{ value | async }}`,
})
export class OutputComponent {
@ngPortalOutput({key: 'foo'}) outputValue: Observable;
}
```### Example: NgModel connection by property name
`ModelComponent` has property `model` with `@ngPortal()` decorator that on change update property `model` in every components with same property and `@ngPortal()` or `@ngPortalOutput()` decorators. eg.:
```ts
import { Component } from '@angular/core';
import { ngPortal } from 'ng-portal';@Component({
selector: 'app-model',
template: ``,
})
export class ModelComponent {
@ngPortal() model: any;
}
```### Example: NgModel connection connected by key
`ModelComponent` has property `model` with `@ngPortal({key: 'foo'})` decorator that on change update every property in every components with `@ngPortal({key: 'foo'})` or `@ngPortalOutput({key: 'foo'})` decorators. eg.:
```ts
import { Component } from '@angular/core';
import { ngPortal, ngPortalOutput } from 'ng-portal';
import { Observable } from 'rxjs';@Component({
selector: 'app-model',
template: ``,
})
export class ModelComponent {
@ngPortal({key: 'foo'}) model: any;
}@Component({
selector: 'app-output',
template: `{{ value | async }}`,
})
export class OutputComponent {
@ngPortalOutput({key: 'foo'}) outputValue: Observable;
}
```## Service
You can inject into your component the `NgPortalService` that expose some utils methods:
```ts
export class NgPortalService {/**
* Send a "value" for the "key" (key or property name)
*/
send(key: string, value: any): void;/**
* Return an Observable for the "key" (key or property name)
*/
get(key: string): Observable;/**
* Return an Observable for all the "key" (key or property name)
*/
getAll(): Observable;
}
```## Support
This is an open-source project. Star this [repository](https://github.com/nigrosimone/ng-portal), if you like it, or even [donate](https://www.paypal.com/paypalme/snwp). Thank you so much!
## My other libraries
I have published some other Angular libraries, take a look:
- [NgSimpleState: Simple state management in Angular with only Services and RxJS](https://www.npmjs.com/package/ng-simple-state)
- [NgHttpCaching: Cache for HTTP requests in Angular application](https://www.npmjs.com/package/ng-http-caching)
- [NgGenericPipe: Generic pipe for Angular application for use a component method into component template.](https://www.npmjs.com/package/ng-generic-pipe)
- [NgLet: Structural directive for sharing data as local variable into html component template](https://www.npmjs.com/package/ng-let)
- [NgForTrackByProperty: Angular global trackBy property directive with strict type checking](https://www.npmjs.com/package/ng-for-track-by-property)