I have a case where an application should return menu based on given context
.
Below are the sources of menuA
and menuB
.
JavaScript
x
28
28
1
// example of menu derived from given source
2
menuA() {
3
return [
4
{
5
a: 'demo1',
6
b: 'demo2',
7
c: 'demo3'
8
},
9
{
10
e: 'demo4',
11
f: 'demo5',
12
g: 'demo6'
13
}
14
];
15
}
16
17
// example of menu fetched from different source eg: db
18
async menuB() {
19
const ret = [];
20
const h = {
21
a: 'demo1',
22
b: 'demo2',
23
c: 'demo3'
24
};
25
ret.push(h);
26
return await ret;
27
}
28
Given my limited knowledge and experience on rxjs
, I was hoping something like the snippet below can accept a string
of menuA
or menuB
to return an observable
of the required menu
:
JavaScript
1
13
13
1
getMenu$(context: string) {
2
return forkJoin(
3
{
4
menuA: of(this.menuA()),
5
menuB: from(this.menuB())
6
}
7
).pipe(
8
mergeMap((m: any) => {
9
return m[context];
10
})
11
)
12
}
13
The above invokes warning as below:
JavaScript
1
4
1
Argument of type 'Observable<unknown>' is not assignable to parameter of type 'OperatorFunction<{ menuA: ({ a: string; b: string; c: string; e?: undefined; f?: undefined; g?: undefined; } | { e: string; f: string; g: string; a?: undefined; b?: undefined; c?: undefined; })[]; menuB: { a: string; b: string; c: string; }[]; }, unknown>'.
2
Type 'Observable<unknown>' provides no match for the signature '(source: Observable<{ menuA: ({ a: string; b: string; c: string; e?: undefined; f?: undefined; g?: undefined; } | { e: string; f: string; g: string; a?: undefined; b?: undefined; c?: undefined; })[]; menuB: { a: string; b: string; c: string; }[]; }>): Observable<...>'.ts(2345)
3
The 'this' context of type 'void' is not assignable to method's 'this' of type 'Observable<any>'.ts(2684)
4
Based on a comment below, I have also added the warning message when map is used to replace the mergeMap:
JavaScript
1
13
13
1
// modified getMenu$:
2
getMenu$(context: string) {
3
return forkJoin(
4
{
5
menuA: of(this.menuA()),
6
menuB: from(this.menuB())
7
}
8
).pipe(
9
map(m => m[context])
10
)
11
}
12
13
Warning message:
JavaScript
1
4
1
Argument of type 'Observable<unknown>' is not assignable to parameter of type 'OperatorFunction<{ menuA: ({ a: string; b: string; c: string; e?: undefined; f?: undefined; g?: undefined; } | { e: string; f: string; g: string; a?: undefined; b?: undefined; c?: undefined; })[]; menuB: { a: string; b: string; c: string; }[]; }, unknown>'.
2
Type 'Observable<unknown>' provides no match for the signature '(source: Observable<{ menuA: ({ a: string; b: string; c: string; e?: undefined; f?: undefined; g?: undefined; } | { e: string; f: string; g: string; a?: undefined; b?: undefined; c?: undefined; })[]; menuB: { a: string; b: string; c: string; }[]; }>): Observable<...>'.ts(2345)
3
The 'this' context of type 'void' is not assignable to method's 'this' of type 'Observable<unknown>'.ts(2684)
4
I am using angular 12
and rxjs "~6.6.0"
.
Advertisement
Answer
DEMO: https://stackblitz.com/edit/typescript-ktm2hz?file=index.ts
JavaScript
1
39
39
1
function menuA() {
2
return [
3
{
4
a: 'demo1',
5
b: 'demo2',
6
c: 'demo3'
7
},
8
{
9
e: 'demo4',
10
f: 'demo5',
11
g: 'demo6'
12
}
13
];
14
}
15
16
// example of menu fetched from different source eg: db
17
function menuB() {
18
return of([
19
{
20
a: 'demo1',
21
b: 'demo2',
22
c: 'demo3'
23
}
24
]).pipe(delay(500));
25
}
26
27
function getMenu$(context: string) {
28
return forkJoin(of(menuA()), menuB().pipe(first())).pipe(
29
map(([menuA, menuB]) => ({ menuA, menuB })),
30
map(m => m[context])
31
);
32
}
33
getMenu$('menuA').subscribe(data => {
34
console.log(data);
35
});
36
getMenu$('menuB').subscribe(data => {
37
console.log(data);
38
});
39
UPDATE:
you need to declare your menus. I have created Enum for it
DEMO with Angular 12: https://stackblitz.com/edit/angular-12-template-wcgohi?file=src/app/app.component.ts
JavaScript
1
59
59
1
import { Component, OnInit } from '@angular/core';
2
import { forkJoin, Observable, of } from 'rxjs';
3
import { first, map, delay } from 'rxjs/operators';
4
5
enum Menus {
6
menuA = 'menuA',
7
menuB = 'menuB'
8
}
9
10
@Component({
11
selector: 'app-root',
12
templateUrl: './app.component.html',
13
styleUrls: ['./app.component.css']
14
})
15
export class AppComponent implements OnInit {
16
menuA(): { [key: string]: any }[] {
17
return [
18
{
19
a: 'demo1',
20
b: 'demo2',
21
c: 'demo3'
22
},
23
{
24
e: 'demo4',
25
f: 'demo5',
26
g: 'demo6'
27
}
28
];
29
}
30
31
// example of menu fetched from different source eg: db
32
menuB(): Observable<{ [key: string]: any }[]> {
33
return of([
34
{
35
a: 'demo1',
36
b: 'demo2',
37
c: 'demo3'
38
}
39
]).pipe(delay(500));
40
}
41
42
getMenu$(context: Menus): Observable<{ [key: string]: any }[]> {
43
return forkJoin([of(this.menuA()), this.menuB().pipe(first())]).pipe(
44
map(([menuA, menuB]) => ({ menuA, menuB })),
45
map(m => m[context])
46
);
47
}
48
ngOnInit() {
49
this.getMenu$(Menus.menuA).subscribe(data => {
50
console.log(data);
51
});
52
this.getMenu$(Menus.menuB).subscribe(data => {
53
console.log(data);
54
});
55
}
56
}
57
58
59