This repository was archived by the owner on Nov 15, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathwizard.component.ts
103 lines (94 loc) · 3.53 KB
/
wizard.component.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/*
* Copyright (c) 2019 Software AG, Darmstadt, Germany and/or its licensors
*
* SPDX-License-Identifier: Apache-2.0
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import {
AfterContentInit,
ChangeDetectorRef,
Component,
ContentChildren,
Input,
OnDestroy,
QueryList
} from "@angular/core";
import {WizardStepComponent} from "./wizard-step.component";
import {BehaviorSubject, Subscription} from "rxjs";
import {AsyncInput} from "@ng-reactive/async-input";
import {filter, startWith} from "rxjs/operators";
/**
* A simple step by step wizard, allows a user to click through a series of different screens
<wizard activeStepId="menu">
<wizard-step stepId="menu">
<button (click)="wizard.selectStep('page1')">Next</button>
</wizard-step>
<wizard-step stepId="page1">
<button (click)="wizard.selectStep('menu')">Back</button>
</wizard-step>
</wizard>
*/
@Component({
selector: 'wizard',
template: `<ng-content></ng-content>`
})
export class WizardComponent implements OnDestroy, AfterContentInit {
private activeStep = new BehaviorSubject<WizardStepComponent | undefined>(undefined);
private stepsById = new Map<string, WizardStepComponent>();
private subscriptions = new Subscription();
@ContentChildren(WizardStepComponent) public steps: QueryList<WizardStepComponent>;
@Input() public activeStepId: string;
@AsyncInput() private activeStepId$ = new BehaviorSubject<string>('');
constructor(private changeDetector: ChangeDetectorRef) {
this.subscriptions.add(
this.activeStepId$
.pipe(filter(() => this.steps != undefined))
.subscribe(stepId => this._selectStep(stepId)));
this.subscriptions.add(
this.activeStep
.subscribe(activeStep => {
Array.from(this.stepsById.values()).filter(step => step != activeStep).forEach(step => step.hidden = true);
if (activeStep != undefined) {
activeStep.hidden = false;
}
})
);
}
ngAfterContentInit(): void {
this.subscriptions.add(this.steps.changes.pipe(startWith({})).subscribe(() => {
this.stepsById = this.steps.reduce((stepsById, step) => {
stepsById.set(step.stepId, step);
return stepsById;
}, new Map<string, WizardStepComponent>());
this._selectStep(this.activeStepId);
}));
}
private _selectStep(stepId: string) {
if (this.stepsById.has(stepId)) {
this.activeStep.next(this.stepsById.get(stepId));
} else {
console.warn(`Could not find step with id: ${stepId}`);
this.activeStep.next(this.steps.first);
}
}
public selectStep(stepId: string) {
this.activeStepId = stepId;
this.activeStepId$.next(stepId);
this.changeDetector.markForCheck();
this.changeDetector.detectChanges();
}
ngOnDestroy(): void {
this.subscriptions.unsubscribe();
}
}