@ringcentral-integration/core
v0.15.0
Published
The foundation package for RingCentral Integration products.
Downloads
120
Readme
@ringcentral-integration/core
This is the foundation package for RingCentral integration products, it is based on Redux and Immer for OOP module model design.
Documentation
Usage
yarn add @ringcentral-integration/core
APIs
RcModule APIs
@ringcentral-integration/core
provides RcModuleV2
base module, decorators state
, action
, computed
, storage
and globalStorage
.
The decorator storage
depends on Storage
Module, And The decorator globalStorage
depends on GlobalStorage
Module.
You should have access to all the dependency modules via
this._deps.fooBar
.
- onInit()
onInit
life cycle for current initialization after all deps modules are all ready.
- override onInitOnce()
onInitOnce
once life cycle for current initialization after all deps modules are all ready.
- onInitSuccess()
onInitSuccess
life cycle for current initialization after this module is ready.
- onReset()
onReset
life cycle for current reset after one of deps modules is not ready.
- onStateChange()
onStateChange
each Redux dispatch action will trigger it once.
For example:
import {
RcModuleV2,
state,
action,
computed,
} from '@ringcentral-integration/core';
class Auth extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
connected = '';
@action
changeConnection(connected) {
this.connected = connected;
}
async connect() {
await this._deps.client.connect();
this.changeConnection(true);
}
@computed((that: Auth) => [that.connected])
get permissions() {
return { writeable: this.connected, readable: true };
}
override async onInitSuccess() {
//
}
}
state
@state
is used to decorate a module state, which is based on the Redux reducer.
action
@action
is used to decorate a method that changes the state of the module (Executing it will dispatch a Redux action), and it does NOT support asynchronous methods.
The method decorated with @action in the current module CANNOT call the method decorated with @action in other modules.
- The @action decorated method should have no side effects.
class ContactsList extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
contacts: Contact[] = [];
@action
addContacts(contacts: Contact[]) { // ❌ bad practice
contacts.forEach(() => {
// fetch avatar
})
// ....
}
override onInitOnce() { // ✅ good practice
watch(
this,
() => this.contacts,
() => {
// fetch avatar
}
)
}
}
- The state operations in the methods decorated by @action should be *mutation updates as possible to ensure patch minimization.
class ContactsList extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
contacts: Contact[] = [];
@action
addContact(contact: Contact) {
// ❌ bad practice
this.contacts = [...this.contacts, contact];
}
@action
addContact(contact: Contact) {
// ✅ good practice
this.contacts.push(contact);
}
}
computed
Use @computed(callback)
, you should make sure that the return value of its callback function is an Array
of dependency collections.
class Auth extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
connected = '';
@state
readable = false;
@computed(({ connected, readable }: Auth) => [connected, readable])
get permissions() {
return { writeable: getWriteable(this.connected), readable: this.readable };
}
}
RcUIModule APIs
@ringcentral-integration/core
provides RcUIModuleV2
base module and all decorators in RcModuleV2
.
For example:
import {
RcUIModuleV2,
computed,
} from '@ringcentral-integration/core';
class DialerUI extends RcUIModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@computed((that: DialerUI) => [
that._deps.dialer.toNumber,
that._deps.call.callType
])
get toNumber() {
if (that._deps.call.callType === 'WebRTC') {
return that._deps.call.toNumber;
}
return this._deps.dialer.toNumber;
}
getUIProps(): DialerUIProps {
return {
toNumber: this.toNumber,
};
}
getUIFunctions(): DialerUIFunctions {
return {
dialout: () => this.dialout(),
};
}
}
Note: RcUIModule should NOT import any React components.
Dependency Injection
In ringcentral-integration/lib/di
, We should reassign constructor
arguments for harmony with RcModuleV2
or RcUIModuleV2
.
Auth.interface.ts
export interface Deps {
alert: Alert;
storage?: Storage;
authOptions?: AuthOptions;
}
Auth.ts
@Module({
name: 'Auth',
deps: [
'Alert',
[{ dep: 'Storage', optional: true }],
[{ dep: 'AuthOptions', optional: true }],
],
})
class Auth extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
}
Note: All module options based on RcModuleV2 have
{ spread: true }
disabled on the DI settings.
Storage and GlobalStorage APIs
Storage
or GlobalStorage
should be injected in module with ringcentral-integration/lib/di
.
And You should pass parameters enableCache
, enableGlobalCache
and storageKey
in constructor
for super
args.
If you only use @storage
, then you only need to pass enableCache
.
If you only use @globalStorage
, then you only need to pass enableGlobalCache
.
For example:
@Module({
name: 'Auth',
deps: [
'Storage',
{ dep: 'AuthOptions', optional: true },
],
})
class Auth extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
enableCache: deps.authOptions?.enableCache ?? true,
enableGlobalCache: deps.authOptions?.enableGlobalCache ?? true,
storageKey: 'Auth',
});
}
@storage
@state
connected = '';
@globalStorage
@state
token = {};
@action
changeConnection(connected) {
this.connected = connected;
}
}
Tracking APIs
Analytics
or AnalyticsOptions
should injection in the factory module, and You can use @track
to decorate a class method.
For example:
@Module({
name: 'Call',
deps: [],
})
class Call extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
// Pass a tracking event type
@track(trackEvents.inbound)
inboundCall() {
//
}
// Pass a function that returns an array `[customTrackEvent, trackProps]`
@track((that: Call, phoneNumber: string) => [
trackEvents.outbound,
{ loginType: that.callType, phoneNumber },
])
async dialout(phoneNumber: string) {
//
}
// Pass a higher-order function and the sub-function has access to the `analytics` module
@track(() => (analytics) => {
analytics.setUserId();
return [trackEvents.authentication];
})
@action
setLoginSuccess(token: TokenInfo) {
//
}
}
State Subscription APIs
watch
It is used to subscribe to some state or @computed
to get the derived computed state, which returns a callback function that can be used to cancel the subscription.
This subscription will only be triggered if the state value of the subscription has been changed.
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
const dispose = watch(
this,
() => this.count,
(newValue, oldValue) => {
// do something
},
);
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
You can pass the option { multiple: true }
, which will support watching multiple values.
For example,
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
const dispose = watch(
this,
() => [this.a, this.b],
(newValue, oldValue) => {
// do something
},
{
multiple: true,
}
);
}
@state
a = 0;
@state
b = 0;
@action
increaseA() {
this.a += 1;
}
@action
increaseB() {
this.b += 1;
}
}
watch
option supports passing inisEqual
function for custom equal.
subscribe
The subscribed function will be triggered after each Redux dispatch action update event. It has a similar mechanism to the RcModuleV2 API onStateChange()
, except that onStateChange()
cannot be unsubscribed, but the unsubscribed function returned by subscribe()
can be used to cancel the subscription.
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
const unsubscribe = watch(
this,
() => {
// do something
},
);
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
createApp
createApp()
is used to boot RcModuleV2-based modules, it requires all module instances to be RcModule V2, it is not compatible modules with RcModule V1. It does not include the dependency injection feature. If you need the API with the dependency injection, please use createApp()
in ringcentral-integration/lib/createApp
.
Example of createApp()
without DI:
import { createApp } from '@ringcentral-integration/core';
class Todo {
@state
list: { text: string; complete: boolean }[] = [];
@action
add(text: string) {
this.list.push({ text, complete: false });
}
}
interface Deps {
todo: Todo
}
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
const unsubscribe = watch(
this,
() => {
// do something
},
);
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
const todo = new Todo();
const counter = new Counter({ todo });
const main = createApp({
main: counter
modules: [todo]
});
Example of createApp()
without DI:
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
const unsubscribe = watch(
this,
() => {
// do something
},
);
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
createApp
createApp()
is used to boot RcModuleV2-based modules, it requires all module instances to be RcModule V2, it is not compatible modules with RcModule V1. It does not include the dependency injection feature. If you need the API with the dependency injection, please use createApp()
in ringcentral-integration/lib/createApp
.
Example of createApp()
without DI:
import { createApp } from '@ringcentral-integration/core';
class Todo {
@state
list: { text: string; complete: boolean }[] = [];
@action
add(text: string) {
this.list.push({ text, complete: false });
}
}
interface Deps {
todo: Todo
}
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
const todo = new Todo();
const counter = new Counter({ todo });
const main = createApp({
main: counter
modules: [todo]
});
Example of createApp()
with DI:
import { createApp } from '@ringcentral-integration/commons/lib/createApp';
@Module({
name: 'Todo',
})
class Todo {
@state
list: { text: string; complete: boolean }[] = [];
@action
add(text: string) {
this.list.push({ text, complete: false });
}
}
interface Deps {
todo: Todo
}
@ModuleFactory({
providers: [
{ provide: 'Todo', useClass: Todo },
],
})
class Counter extends RcModuleV2<Deps> {
constructor(deps: Deps) {
super({
deps,
});
}
@state
count = 0;
@action
increase() {
this.count += 1;
}
}
const main = createApp(Counter);
Debugging
_depsCheck()
It is used to check the readiness of dependent modules in depth, and it will return modules whose dependency modules are all ready but not ready themselves.
_changeState(callback)
It can be used to temporarily modify any module state, and its argument should be a callback function that modifies the state.