mirror of
https://github.com/zitadel/zitadel.git
synced 2025-02-28 16:17:23 +00:00
feat(console): limited actions (#3164)
* max count features * deactivate, activate * actions, limited * disable without permission, show action state in flow Co-authored-by: Livio Amstutz <livio.a@gmail.com>
This commit is contained in:
parent
78af86db98
commit
3bf9adece5
@ -265,6 +265,8 @@
|
||||
</mat-slide-toggle>
|
||||
</div>
|
||||
|
||||
<p class="feature-section">{{'FEATURES.HEADERS.FLOWS' | translate}}</p>
|
||||
|
||||
<div class="row">
|
||||
<div class="featureavatar pink">
|
||||
<i class="icon las la-exchange-alt"></i>
|
||||
@ -272,9 +274,25 @@
|
||||
<span class="left-desc">{{'FEATURES.DATA.FLOWS' | translate}}</span>
|
||||
<span class="fill-space"></span>
|
||||
<template [ngTemplateOutlet]="templateRef" [ngTemplateOutletContext]="{active: features.actions}"></template>
|
||||
<mat-slide-toggle class="toggle" color="primary" name="hasNumber" ngDefaultControl [(ngModel)]="features.actions"
|
||||
*ngIf="(['iam.features.write'] | hasRole | async)">
|
||||
</mat-slide-toggle>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<span class="fill-space"></span>
|
||||
<cnsl-form-field class="flow-select">
|
||||
<cnsl-label>{{ 'FEATURES.DATA.FLOW.TYPE' | translate }}</cnsl-label>
|
||||
<mat-select [(ngModel)]="features.actionsAllowed"
|
||||
[disabled]="(['iam.features.write'] | hasRole | async) === false">
|
||||
<mat-option *ngFor="let allowedType of actionsSelection" [value]="allowedType">
|
||||
{{ 'FEATURES.DATA.FLOW.ACTIONSALLOWED.'+allowedType | translate}}
|
||||
</mat-option>
|
||||
</mat-select>
|
||||
</cnsl-form-field>
|
||||
|
||||
<cnsl-form-field *ngIf="features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX" class="flow-count">
|
||||
<cnsl-label>{{ 'FEATURES.DATA.FLOW.COUNT' | translate }}</cnsl-label>
|
||||
<input cnslInput type="number" [(ngModel)]="features.maxActions"
|
||||
[disabled]="(['iam.features.write'] | hasRole | async) === false" ngDefaultControl />
|
||||
</cnsl-form-field>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
@ -17,10 +17,10 @@
|
||||
.title {
|
||||
font-size: 14px;
|
||||
color: var(--grey);
|
||||
margin-bottom: .5rem;
|
||||
margin-bottom: 0.5rem;
|
||||
|
||||
a {
|
||||
margin-left: .5rem;
|
||||
margin-left: 0.5rem;
|
||||
cursor: pointer;
|
||||
}
|
||||
}
|
||||
@ -30,19 +30,19 @@
|
||||
align-items: center;
|
||||
|
||||
a {
|
||||
margin-left: .5rem;
|
||||
margin-left: 0.5rem;
|
||||
}
|
||||
}
|
||||
|
||||
img {
|
||||
height: 15px;
|
||||
width: auto;
|
||||
margin-left: .5rem;
|
||||
margin-left: 0.5rem;
|
||||
}
|
||||
}
|
||||
|
||||
.spinner {
|
||||
margin: .5rem;
|
||||
margin: 0.5rem;
|
||||
}
|
||||
|
||||
.error {
|
||||
@ -60,8 +60,8 @@
|
||||
height: 1px;
|
||||
width: 100%;
|
||||
background-color: var(--grey);
|
||||
opacity: .5;
|
||||
margin: .5rem 0;
|
||||
opacity: 0.5;
|
||||
margin: 0.5rem 0;
|
||||
display: block;
|
||||
}
|
||||
|
||||
@ -80,7 +80,7 @@
|
||||
.row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: .3rem 0;
|
||||
padding: 0.3rem 0;
|
||||
|
||||
.featureavatar {
|
||||
margin-right: 1rem;
|
||||
@ -137,7 +137,7 @@
|
||||
}
|
||||
|
||||
.left-desc {
|
||||
font-size: .9rem;
|
||||
font-size: 0.9rem;
|
||||
margin-right: 1rem;
|
||||
}
|
||||
|
||||
@ -149,6 +149,18 @@
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.flow-select {
|
||||
flex-shrink: 1;
|
||||
min-width: 150px;
|
||||
margin-left: 1rem;
|
||||
}
|
||||
|
||||
.flow-count {
|
||||
flex-shrink: 1;
|
||||
width: 70px;
|
||||
margin-left: 1rem;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -8,7 +8,7 @@ import {
|
||||
SetDefaultFeaturesRequest,
|
||||
SetOrgFeaturesRequest,
|
||||
} from 'src/app/proto/generated/zitadel/admin_pb';
|
||||
import { Features } from 'src/app/proto/generated/zitadel/features_pb';
|
||||
import { ActionsAllowed, Features } from 'src/app/proto/generated/zitadel/features_pb';
|
||||
import { GetFeaturesResponse } from 'src/app/proto/generated/zitadel/management_pb';
|
||||
import { Org } from 'src/app/proto/generated/zitadel/org_pb';
|
||||
import { AdminService } from 'src/app/services/admin.service';
|
||||
@ -46,6 +46,13 @@ export class FeaturesComponent implements OnDestroy {
|
||||
public stripeURL: string = '';
|
||||
public stripeCustomer!: StripeCustomer;
|
||||
|
||||
public actionsSelection: any = [
|
||||
ActionsAllowed.ACTIONS_ALLOWED_NOT_ALLOWED,
|
||||
ActionsAllowed.ACTIONS_ALLOWED_MAX,
|
||||
ActionsAllowed.ACTIONS_ALLOWED_UNLIMITED,
|
||||
];
|
||||
public ActionsAllowed: any = ActionsAllowed;
|
||||
|
||||
constructor(
|
||||
private route: ActivatedRoute,
|
||||
private toast: ToastService,
|
||||
@ -60,27 +67,32 @@ export class FeaturesComponent implements OnDestroy {
|
||||
if (temporg) {
|
||||
this.org = temporg;
|
||||
}
|
||||
this.sub = this.route.data.pipe(switchMap(data => {
|
||||
this.serviceType = data.serviceType;
|
||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||
this.managementService = this.injector.get(ManagementService as Type<ManagementService>);
|
||||
}
|
||||
return this.route.params;
|
||||
})).subscribe(_ => {
|
||||
this.fetchData();
|
||||
});
|
||||
this.sub = this.route.data
|
||||
.pipe(
|
||||
switchMap((data) => {
|
||||
this.serviceType = data.serviceType;
|
||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||
this.managementService = this.injector.get(ManagementService as Type<ManagementService>);
|
||||
}
|
||||
return this.route.params;
|
||||
}),
|
||||
)
|
||||
.subscribe((_) => {
|
||||
this.fetchData();
|
||||
});
|
||||
|
||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||
this.customerLoading = true;
|
||||
this.subService.getCustomer(this.org.id)
|
||||
.then(payload => {
|
||||
this.subService
|
||||
.getCustomer(this.org.id)
|
||||
.then((payload) => {
|
||||
this.customerLoading = false;
|
||||
this.stripeCustomer = payload;
|
||||
if (this.customerValid) {
|
||||
this.getLinkToStripe();
|
||||
}
|
||||
})
|
||||
.catch(error => {
|
||||
.catch((error) => {
|
||||
this.customerLoading = false;
|
||||
console.error(error);
|
||||
});
|
||||
@ -99,13 +111,16 @@ export class FeaturesComponent implements OnDestroy {
|
||||
width: '400px',
|
||||
});
|
||||
|
||||
dialogRefPhone.afterClosed().subscribe(customer => {
|
||||
dialogRefPhone.afterClosed().subscribe((customer) => {
|
||||
if (customer) {
|
||||
console.log(customer);
|
||||
this.stripeCustomer = customer;
|
||||
this.subService.setCustomer(this.org.id, customer).then(() => {
|
||||
this.getLinkToStripe();
|
||||
}).catch(console.error);
|
||||
this.subService
|
||||
.setCustomer(this.org.id, customer)
|
||||
.then(() => {
|
||||
this.getLinkToStripe();
|
||||
})
|
||||
.catch(console.error);
|
||||
}
|
||||
});
|
||||
}
|
||||
@ -113,12 +128,13 @@ export class FeaturesComponent implements OnDestroy {
|
||||
public getLinkToStripe(): void {
|
||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||
this.stripeLoading = true;
|
||||
this.subService.getLink(this.org.id, window.location.href)
|
||||
.then(payload => {
|
||||
this.subService
|
||||
.getLink(this.org.id, window.location.href)
|
||||
.then((payload) => {
|
||||
this.stripeLoading = false;
|
||||
this.stripeURL = payload.redirect_url;
|
||||
})
|
||||
.catch(error => {
|
||||
.catch((error) => {
|
||||
this.stripeLoading = false;
|
||||
console.error(error);
|
||||
});
|
||||
@ -126,7 +142,7 @@ export class FeaturesComponent implements OnDestroy {
|
||||
}
|
||||
|
||||
public fetchData(): void {
|
||||
this.getData().then(resp => {
|
||||
this.getData().then((resp) => {
|
||||
if (resp?.features) {
|
||||
this.features = resp.features;
|
||||
}
|
||||
@ -167,13 +183,18 @@ export class FeaturesComponent implements OnDestroy {
|
||||
req.setPrivacyPolicy(this.features.privacyPolicy);
|
||||
req.setMetadataUser(this.features.metadataUser);
|
||||
req.setLockoutPolicy(this.features.lockoutPolicy);
|
||||
req.setActions(this.features.actions);
|
||||
// req.setActions(this.features.actions);
|
||||
req.setActionsAllowed(this.features.actionsAllowed);
|
||||
req.setMaxActions(this.features.maxActions);
|
||||
|
||||
this.adminService.setOrgFeatures(req).then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||
}).catch(error => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
this.adminService
|
||||
.setOrgFeatures(req)
|
||||
.then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||
})
|
||||
.catch((error) => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
break;
|
||||
case FeatureServiceType.ADMIN:
|
||||
// update Default org iam policy?
|
||||
@ -192,27 +213,35 @@ export class FeaturesComponent implements OnDestroy {
|
||||
dreq.setCustomTextMessage(this.features.customTextMessage);
|
||||
dreq.setMetadataUser(this.features.metadataUser);
|
||||
dreq.setLockoutPolicy(this.features.lockoutPolicy);
|
||||
dreq.setActions(this.features.actions);
|
||||
// dreq.setActions(this.features.actions);
|
||||
dreq.setActionsAllowed(this.features.actionsAllowed);
|
||||
dreq.setMaxActions(this.features.maxActions);
|
||||
|
||||
this.adminService.setDefaultFeatures(dreq).then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||
}).catch(error => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
this.adminService
|
||||
.setDefaultFeatures(dreq)
|
||||
.then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||
})
|
||||
.catch((error) => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
public resetFeatures(): void {
|
||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||
this.adminService.resetOrgFeatures(this.org.id).then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true);
|
||||
setTimeout(() => {
|
||||
this.fetchData();
|
||||
}, 1000);
|
||||
}).catch(error => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
this.adminService
|
||||
.resetOrgFeatures(this.org.id)
|
||||
.then(() => {
|
||||
this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true);
|
||||
setTimeout(() => {
|
||||
this.fetchData();
|
||||
}, 1000);
|
||||
})
|
||||
.catch((error) => {
|
||||
this.toast.showError(error);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@ -225,13 +254,15 @@ export class FeaturesComponent implements OnDestroy {
|
||||
}
|
||||
|
||||
get customerValid(): boolean {
|
||||
return !!this.stripeCustomer?.contact &&
|
||||
return (
|
||||
!!this.stripeCustomer?.contact &&
|
||||
!!this.stripeCustomer?.address &&
|
||||
!!this.stripeCustomer?.city &&
|
||||
!!this.stripeCustomer?.postal_code;
|
||||
!!this.stripeCustomer?.postal_code
|
||||
);
|
||||
}
|
||||
|
||||
get customerCountry(): Country | undefined {
|
||||
return COUNTRIES.find(country => country.isoCode === this.stripeCustomer.country);
|
||||
return COUNTRIES.find((country) => country.isoCode === this.stripeCustomer.country);
|
||||
}
|
||||
}
|
||||
|
@ -1,10 +1,19 @@
|
||||
<cnsl-refresh-table [loading]="loading$ | async" (refreshed)="refreshPage()" [dataSize]="dataSource?.data?.length ?? 0"
|
||||
[timestamp]="actionsResult?.details?.viewTimestamp" [selection]="selection">
|
||||
<div actions>
|
||||
<div actions *ngIf="selection.isEmpty()">
|
||||
<a color="primary" mat-raised-button (click)="openAddAction()">
|
||||
<mat-icon class="icon">add</mat-icon>{{ 'ACTIONS.NEW' | translate }}
|
||||
</a>
|
||||
</div>
|
||||
<div actions *ngIf="!selection.isEmpty()">
|
||||
<button class="action-state-btn" mat-stroked-button (click)="deactivateSelection()">
|
||||
{{ 'ACTIONS.DEACTIVATE' | translate }}
|
||||
</button>
|
||||
<button class="action-state-btn" mat-stroked-button (click)="activateSelection()">
|
||||
{{ 'ACTIONS.REACTIVATE' | translate }}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="table-wrapper">
|
||||
<table class="table" mat-table [dataSource]="dataSource">
|
||||
|
@ -1,3 +1,6 @@
|
||||
.action-state-btn {
|
||||
margin-left: 0.5rem;
|
||||
}
|
||||
|
||||
.table-wrapper {
|
||||
overflow: auto;
|
||||
|
@ -137,4 +137,40 @@ export class ActionTableComponent implements OnInit {
|
||||
public refreshPage(): void {
|
||||
this.getData(this.paginator.pageSize, this.paginator.pageIndex * this.paginator.pageSize);
|
||||
}
|
||||
|
||||
public deactivateSelection(): Promise<void> {
|
||||
const prom = this.selection.selected.map((action) => {
|
||||
return this.mgmtService.deactivateAction(action.id);
|
||||
});
|
||||
|
||||
return Promise.all(prom)
|
||||
.then(() => {
|
||||
this.selection.clear();
|
||||
this.toast.showInfo('FLOWS.TOAST.ACTIONDEACTIVATED', true);
|
||||
this.getData(10, 0);
|
||||
})
|
||||
.catch((error) => {
|
||||
this.selection.clear();
|
||||
this.toast.showError(error);
|
||||
this.getData(10, 0);
|
||||
});
|
||||
}
|
||||
|
||||
public activateSelection(): Promise<void> {
|
||||
const prom = this.selection.selected.map((action) => {
|
||||
return this.mgmtService.reactivateAction(action.id);
|
||||
});
|
||||
|
||||
return Promise.all(prom)
|
||||
.then(() => {
|
||||
this.selection.clear();
|
||||
this.toast.showInfo('FLOWS.TOAST.ACTIONREACTIVATED', true);
|
||||
this.getData(10, 0);
|
||||
})
|
||||
.catch((error) => {
|
||||
this.selection.clear();
|
||||
this.toast.showError(error);
|
||||
this.getData(10, 0);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
@ -2,6 +2,9 @@
|
||||
<h1>{{ 'FLOWS.TITLE' | translate }}</h1>
|
||||
<p class="desc">{{'FLOWS.DESCRIPTION' | translate }}</p>
|
||||
|
||||
<cnsl-info-section class="max-actions" *ngIf="maxActions">{{'FLOWS.ACTIONSMAX' | translate: ({value: maxActions}) }}
|
||||
</cnsl-info-section>
|
||||
|
||||
<cnsl-info-section *ngIf="(['actions'] | hasFeature | async) === false" [featureLink]="['/org/features']" class="info"
|
||||
[type]="InfoSectionType.WARN">
|
||||
<span [innerHTML]="'FEATURES.NOTAVAILABLE' | translate: ({value: 'actions'})"></span>
|
||||
@ -44,11 +47,15 @@
|
||||
<mat-icon svgIcon="mdi_arrow_right_bottom" class="icon"></mat-icon>
|
||||
<span>{{'FLOWS.TRIGGERTYPES.'+trigger.triggerType | translate}}</span>
|
||||
<span class="fill-space"></span>
|
||||
<div class="action-wrapper" cdkDropList (cdkDropListDropped)="drop(i, trigger.actionsList, $event)">
|
||||
<div cdkDrag cdkDragLockAxis="y" cdkDragBoundary=".action-wrapper" class="action"
|
||||
<div class="flow-action-wrapper" cdkDropList (cdkDropListDropped)="drop(i, trigger.actionsList, $event)">
|
||||
<div cdkDrag cdkDragLockAxis="y" cdkDragBoundary=".action-wrapper" class="flow-action"
|
||||
*ngFor="let action of trigger.actionsList">
|
||||
<i class="las la-code"></i>
|
||||
<span>{{action.name}}</span>
|
||||
<span class="flow-action-name">{{action.name}}</span>
|
||||
<span class="fill-space"></span>
|
||||
<span class="state"
|
||||
[ngClass]="{'active': action.state === ActionState.ACTION_STATE_ACTIVE,'inactive': action.state === ActionState.ACTION_STATE_INACTIVE }">
|
||||
{{'FLOWS.STATES.'+action.state | translate}}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -18,7 +18,7 @@ h1 {
|
||||
}
|
||||
|
||||
i {
|
||||
margin-left: .5rem;
|
||||
margin-left: 0.5rem;
|
||||
}
|
||||
}
|
||||
|
||||
@ -29,8 +29,8 @@ h1 {
|
||||
|
||||
.flow-type {
|
||||
padding: 1rem 1rem;
|
||||
margin: .5rem 0;
|
||||
border-radius: .5rem;
|
||||
margin: 0.5rem 0;
|
||||
border-radius: 0.5rem;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
@ -39,7 +39,7 @@ h1 {
|
||||
.topelements {
|
||||
border: 3px solid var(--color-main);
|
||||
border-radius: 1rem;
|
||||
padding: 0 .5rem;
|
||||
padding: 0 0.5rem;
|
||||
}
|
||||
|
||||
.trigger-wrapper {
|
||||
@ -57,13 +57,13 @@ h1 {
|
||||
}
|
||||
|
||||
.trigger {
|
||||
padding: .5rem 1rem;
|
||||
border-radius: .5rem;
|
||||
padding: 0.5rem 1rem;
|
||||
border-radius: 0.5rem;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
background: var(--color-main);
|
||||
color: white;
|
||||
margin: .5rem 0;
|
||||
margin: 0.5rem 0;
|
||||
min-height: 40px;
|
||||
|
||||
.icon {
|
||||
@ -74,18 +74,27 @@ h1 {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.action-wrapper {
|
||||
padding: 0 .5rem;
|
||||
.flow-action-wrapper {
|
||||
padding: 0 0.5rem;
|
||||
margin: 0;
|
||||
|
||||
.action {
|
||||
.flow-action {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
font-size: 14px;
|
||||
padding: .5rem 0;
|
||||
padding: 0.5rem 0;
|
||||
cursor: move;
|
||||
|
||||
.flow-action-name {
|
||||
margin-right: 1rem;
|
||||
}
|
||||
|
||||
.fill-space {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
i {
|
||||
margin-right: .5rem;
|
||||
margin-right: 0.5rem;
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -105,13 +114,13 @@ h1 {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
font-size: 14px;
|
||||
border-radius: .5rem;
|
||||
padding: 0 .5rem;
|
||||
border-radius: 0.5rem;
|
||||
padding: 0 0.5rem;
|
||||
background-color: var(--color-main);
|
||||
box-shadow: 0 5px 5px -3px rgba(0, 0, 0, .2), 0 8px 10px 1px rgba(0, 0, 0, .14), 0 3px 14px 2px rgba(0, 0, 0, .12);
|
||||
box-shadow: 0 5px 5px -3px rgba(0, 0, 0, 0.2), 0 8px 10px 1px rgba(0, 0, 0, 0.14), 0 3px 14px 2px rgba(0, 0, 0, 0.12);
|
||||
|
||||
i {
|
||||
margin-right: .5rem;
|
||||
margin-right: 0.5rem;
|
||||
}
|
||||
}
|
||||
|
||||
@ -120,5 +129,5 @@ h1 {
|
||||
}
|
||||
|
||||
.cdk-drag-animating {
|
||||
transition: transform 250ms cubic-bezier(0, 0, .2, 1);
|
||||
transition: transform 250ms cubic-bezier(0, 0, 0.2, 1);
|
||||
}
|
||||
|
@ -4,7 +4,8 @@ import { FormControl } from '@angular/forms';
|
||||
import { MatDialog } from '@angular/material/dialog';
|
||||
import { InfoSectionType } from 'src/app/modules/info-section/info-section.component';
|
||||
import { WarnDialogComponent } from 'src/app/modules/warn-dialog/warn-dialog.component';
|
||||
import { Action, Flow, FlowType, TriggerType } from 'src/app/proto/generated/zitadel/action_pb';
|
||||
import { Action, ActionState, Flow, FlowType, TriggerType } from 'src/app/proto/generated/zitadel/action_pb';
|
||||
import { ActionsAllowed } from 'src/app/proto/generated/zitadel/features_pb';
|
||||
import { SetTriggerActionsRequest } from 'src/app/proto/generated/zitadel/management_pb';
|
||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||
import { ToastService } from 'src/app/services/toast.service';
|
||||
@ -26,14 +27,28 @@ export class ActionsComponent {
|
||||
|
||||
public selection: Action.AsObject[] = [];
|
||||
public InfoSectionType: any = InfoSectionType;
|
||||
public maxActions: number | null = null;
|
||||
public ActionState: any = ActionState;
|
||||
|
||||
constructor(private mgmtService: ManagementService, private dialog: MatDialog, private toast: ToastService) {
|
||||
this.mgmtService.getFeatures().then((featuresResp) => {
|
||||
if (featuresResp && featuresResp.features) {
|
||||
const features = featuresResp.features;
|
||||
this.maxActions =
|
||||
features && features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX
|
||||
? features.maxActions
|
||||
: features.actionsAllowed === ActionsAllowed.ACTIONS_ALLOWED_MAX
|
||||
? null
|
||||
: 0;
|
||||
}
|
||||
});
|
||||
this.loadFlow();
|
||||
}
|
||||
|
||||
private loadFlow() {
|
||||
this.mgmtService.getFlow(this.flowType).then((flowResponse) => {
|
||||
if (flowResponse.flow) this.flow = flowResponse.flow;
|
||||
console.log(this.flow);
|
||||
});
|
||||
}
|
||||
|
||||
@ -92,7 +107,6 @@ export class ActionsComponent {
|
||||
}
|
||||
|
||||
saveFlow(index: number) {
|
||||
console.log(this.flow.triggerActionsList[index].actionsList.map((action) => action.id));
|
||||
this.mgmtService
|
||||
.setTriggerActions(
|
||||
this.flow.triggerActionsList[index].actionsList.map((action) => action.id),
|
||||
|
@ -74,6 +74,8 @@ import {
|
||||
ClearFlowResponse,
|
||||
CreateActionRequest,
|
||||
CreateActionResponse,
|
||||
DeactivateActionRequest,
|
||||
DeactivateActionResponse,
|
||||
DeactivateAppRequest,
|
||||
DeactivateAppResponse,
|
||||
DeactivateOrgIDPRequest,
|
||||
@ -238,6 +240,8 @@ import {
|
||||
ListUserMetadataResponse,
|
||||
ListUsersRequest,
|
||||
ListUsersResponse,
|
||||
ReactivateActionRequest,
|
||||
ReactivateActionResponse,
|
||||
ReactivateAppRequest,
|
||||
ReactivateAppResponse,
|
||||
ReactivateOrgIDPRequest,
|
||||
@ -919,6 +923,18 @@ export class ManagementService {
|
||||
return this.grpcService.mgmt.deleteAction(req, null).then((resp) => resp.toObject());
|
||||
}
|
||||
|
||||
public deactivateAction(id: string): Promise<DeactivateActionResponse.AsObject> {
|
||||
const req = new DeactivateActionRequest();
|
||||
req.setId(id);
|
||||
return this.grpcService.mgmt.deactivateAction(req, null).then((resp) => resp.toObject());
|
||||
}
|
||||
|
||||
public reactivateAction(id: string): Promise<ReactivateActionResponse.AsObject> {
|
||||
const req = new ReactivateActionRequest();
|
||||
req.setId(id);
|
||||
return this.grpcService.mgmt.reactivateAction(req, null).then((resp) => resp.toObject());
|
||||
}
|
||||
|
||||
public listActions(
|
||||
limit?: number,
|
||||
offset?: number,
|
||||
|
@ -564,6 +564,7 @@
|
||||
"FLOWTYPE": "Flow Typ",
|
||||
"TRIGGERTYPE": "Trigger Typ",
|
||||
"ACTIONS": "Aktionen",
|
||||
"ACTIONSMAX": "Basierend auf Ihrem Tier steht Ihnen eine begrenzte Anzahl von Aktionen ({{value}}) zur Verfügung. Stellen Sie sicher, dass Sie diejenigen deaktivieren, die Sie nicht benötigen, oder erwägen Sie ein Upgrade.",
|
||||
"DIALOG": {
|
||||
"ADD": {
|
||||
"TITLE": "Aktion erstellen"
|
||||
@ -581,7 +582,9 @@
|
||||
}
|
||||
},
|
||||
"TOAST": {
|
||||
"ACTIONSSET": "Aktionen gesetzt"
|
||||
"ACTIONSSET": "Aktionen gesetzt",
|
||||
"ACTIONREACTIVATED": "Aktionen erfolgreich reaktiviert",
|
||||
"ACTIONDEACTIVATED": "Aktionen erfolgreich deaktiviert"
|
||||
}
|
||||
},
|
||||
"IAM": {
|
||||
@ -756,7 +759,16 @@
|
||||
"CUSTOMTEXTMESSAGE": "Benutzerdefinierte Benachrichtigungstexte",
|
||||
"PRIVACYPOLICY": "Benutzerdefinierte Datenschutzrichtlinie und AGB",
|
||||
"METADATAUSER": "User Metadata",
|
||||
"FLOWS": "Aktionen und Abläufe"
|
||||
"FLOWS": "Aktionen und Abläufe",
|
||||
"FLOW": {
|
||||
"ACTIONSALLOWED": {
|
||||
"0": "Nicht erlaubt",
|
||||
"1": "Limitierte Aktionen",
|
||||
"2": "Unlimitiert"
|
||||
},
|
||||
"TYPE": "Verwendungsart",
|
||||
"COUNT": "Anzahl"
|
||||
}
|
||||
},
|
||||
"TIERSTATES": {
|
||||
"0": "Aktiv",
|
||||
|
@ -564,6 +564,7 @@
|
||||
"FLOWTYPE": "Flow Type",
|
||||
"TRIGGERTYPE": "Trigger Type",
|
||||
"ACTIONS": "Actions",
|
||||
"ACTIONSMAX": "Based on your Tier, you have available a limited Number of Actions ({{value}}). Make sure to deaktivate those you are not in need or consider upgrading your tier.",
|
||||
"DIALOG": {
|
||||
"ADD": {
|
||||
"TITLE": "Create an Action"
|
||||
@ -581,7 +582,9 @@
|
||||
}
|
||||
},
|
||||
"TOAST": {
|
||||
"ACTIONSSET": "Actions set"
|
||||
"ACTIONSSET": "Actions set",
|
||||
"ACTIONREACTIVATED": "Actions reactivated with success",
|
||||
"ACTIONDEACTIVATED": "Actions deactivated with success"
|
||||
}
|
||||
},
|
||||
"IAM": {
|
||||
@ -756,7 +759,16 @@
|
||||
"CUSTOMTEXTMESSAGE": "Custom notification mail texts",
|
||||
"PRIVACYPOLICY": "Custom Privacy Policy and TOS Links",
|
||||
"METADATAUSER": "User Metadata",
|
||||
"FLOWS": "Actions and Flows"
|
||||
"FLOWS": "Actions and Flows",
|
||||
"FLOW": {
|
||||
"ACTIONSALLOWED": {
|
||||
"0": "Not allowed",
|
||||
"1": "Limited Actions",
|
||||
"2": "Unlimited Actions"
|
||||
},
|
||||
"TYPE": "Type of use",
|
||||
"COUNT": "Number"
|
||||
}
|
||||
},
|
||||
"TIERSTATES": {
|
||||
"0": "Active",
|
||||
|
@ -564,6 +564,7 @@
|
||||
"FLOWTYPE": "Tipo processo",
|
||||
"TRIGGERTYPE": "Tipo trigger",
|
||||
"ACTIONS": "Azioni",
|
||||
"ACTIONSMAX": "In base al tuo tier, hai a disposizione un numero limitato di azioni ({{value}}). Assicurati di disattivare quelli di cui non hai bisogno o considera di fare un upgrade.",
|
||||
"DIALOG": {
|
||||
"ADD": {
|
||||
"TITLE": "Crea azione"
|
||||
@ -581,7 +582,9 @@
|
||||
}
|
||||
},
|
||||
"TOAST": {
|
||||
"ACTIONSSET": "Azioni salvate!"
|
||||
"ACTIONSSET": "Azioni salvate!",
|
||||
"ACTIONREACTIVATED": "Azioni riattivati con successo",
|
||||
"ACTIONDEACTIVATED": "Azioni disattivati con successo"
|
||||
}
|
||||
},
|
||||
"IAM": {
|
||||
@ -756,7 +759,16 @@
|
||||
"CUSTOMTEXTMESSAGE": "Testi email personalizzati",
|
||||
"PRIVACYPOLICY": "Link personalizzati all'informativa sulla privacy e ai TOS",
|
||||
"METADATAUSER": "Metadati utente",
|
||||
"FLOWS": "Azioni e processi"
|
||||
"FLOWS": "Azioni e processi",
|
||||
"FLOW": {
|
||||
"ACTIONSALLOWED": {
|
||||
"0": "Non abilitato",
|
||||
"1": "Numero limitato",
|
||||
"2": "Illimitato"
|
||||
},
|
||||
"COUNT": "Anzahl",
|
||||
"TYPE": "Tipo di utilizzo"
|
||||
}
|
||||
},
|
||||
"TIERSTATES": {
|
||||
"0": "Attivo",
|
||||
|
Loading…
x
Reference in New Issue
Block a user