feat(console): stripe (#1590)
* prepare request for sub service * show detail, i18n * form dialog * country * stripe button * feature * flex * rm logs * org creationdate, lint * rm log
@ -25,6 +25,7 @@ import { AuthConfig, OAuthModule, OAuthStorage } from 'angular-oauth2-oidc';
|
|||||||
import { QuicklinkModule } from 'ngx-quicklink';
|
import { QuicklinkModule } from 'ngx-quicklink';
|
||||||
import { OnboardingModule } from 'src/app/modules/onboarding/onboarding.module';
|
import { OnboardingModule } from 'src/app/modules/onboarding/onboarding.module';
|
||||||
import { RegExpPipeModule } from 'src/app/pipes/regexp-pipe/regexp-pipe.module';
|
import { RegExpPipeModule } from 'src/app/pipes/regexp-pipe/regexp-pipe.module';
|
||||||
|
import { SubscriptionService } from 'src/app/services/subscription.service';
|
||||||
|
|
||||||
import { environment } from '../environments/environment';
|
import { environment } from '../environments/environment';
|
||||||
import { AppRoutingModule } from './app-routing.module';
|
import { AppRoutingModule } from './app-routing.module';
|
||||||
@ -178,6 +179,7 @@ const authConfig: AuthConfig = {
|
|||||||
GrpcService,
|
GrpcService,
|
||||||
AuthenticationService,
|
AuthenticationService,
|
||||||
GrpcAuthService,
|
GrpcAuthService,
|
||||||
|
SubscriptionService,
|
||||||
{ provide: 'windowObject', useValue: window },
|
{ provide: 'windowObject', useValue: window },
|
||||||
],
|
],
|
||||||
bootstrap: [AppComponent],
|
bootstrap: [AppComponent],
|
||||||
|
1539
console/src/app/modules/features/country.ts
Normal file
@ -1,6 +1,39 @@
|
|||||||
<app-detail-layout [backRouterLink]="[ serviceType === FeatureServiceType.ADMIN ? '/iam/policies' : '/org']"
|
<app-detail-layout [backRouterLink]="[ serviceType === FeatureServiceType.ADMIN ? '/iam/policies' : '/org']"
|
||||||
[title]="'ZITADEL ' +features?.tier.name + ' ' + ('FEATURES.TITLE' | translate)" [description]="'FEATURES.DESCRIPTION' | translate">
|
[title]="('FEATURES.TITLE' | translate)" [description]="'FEATURES.DESCRIPTION' | translate">
|
||||||
<p class="default" *ngIf="isDefault"> {{'POLICY.DEFAULTLABEL' | translate}}</p>
|
|
||||||
|
<h2>{{'FEATURES.TIER.TITLE' | translate}}</h2>
|
||||||
|
<p *ngIf="serviceType === FeatureServiceType.MGMT" class="tier-desc">{{'FEATURES.TIER.DESCRIPTION' | translate}}
|
||||||
|
{{'FEATURES.TIER.QUESTIONS' | translate}} <a href="mailto:support@zitadel.ch">support@zitadel.ch</a>.</p>
|
||||||
|
|
||||||
|
<div class="detail">
|
||||||
|
<p class="title">{{'FEATURES.TIER.NAME' | translate}}</p>
|
||||||
|
<p>{{features?.tier?.name}}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ng-container *ngIf="serviceType === FeatureServiceType.MGMT">
|
||||||
|
<mat-spinner class="spinner" diameter="20" *ngIf="customerLoading || stripeLoading"></mat-spinner>
|
||||||
|
<div class="detail" *ngIf="stripeCustomer">
|
||||||
|
<p class="title">{{'FEATURES.TIER.DETAILS' | translate}}
|
||||||
|
<a (click)="updateCustomer()">{{'ACTIONS.EDIT' | translate}}</a>
|
||||||
|
</p>
|
||||||
|
<p>{{stripeCustomer?.contact}}</p>
|
||||||
|
<p *ngIf="stripeCustomer.company">{{stripeCustomer?.company}}</p>
|
||||||
|
<p>{{stripeCustomer?.address}}</p>
|
||||||
|
<p *ngIf="stripeCustomer?.postal_code || stripeCustomer?.city || stripeCustomer?.country">
|
||||||
|
{{stripeCustomer?.postal_code}} {{stripeCustomer?.city}} {{stripeCustomer?.country}}
|
||||||
|
<img *ngIf="customerCountry" height="20px" width="30px"
|
||||||
|
style="margin-right: 1rem; border-radius: 2px; vertical-align: middle;"
|
||||||
|
src="../../../assets/flags/{{customerCountry.isoCode.toLowerCase()}}.png" />
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="error" *ngIf="stripeCustomer && !customerValid">{{'FEATURES.TIER.CUSTOMERINVALID' | translate}}</p>
|
||||||
|
|
||||||
|
<div class="current-tier">
|
||||||
|
<a [disabled]="!org.id || !customerValid" mat-raised-button [href]="stripeURL" target="_blank"
|
||||||
|
alt="change tier">{{'FEATURES.TIER.BTN' | translate}}</a>
|
||||||
|
</div>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
<ng-template appHasRole [appHasRole]="['iam.features.delete']">
|
<ng-template appHasRole [appHasRole]="['iam.features.delete']">
|
||||||
<button *ngIf="serviceType === FeatureServiceType.MGMT && !isDefault"
|
<button *ngIf="serviceType === FeatureServiceType.MGMT && !isDefault"
|
||||||
@ -9,12 +42,15 @@
|
|||||||
</button>
|
</button>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
|
||||||
<div class="content" *ngIf="features">
|
<div class="divider"></div>
|
||||||
|
|
||||||
|
<p class="default" *ngIf="isDefault"> {{'POLICY.DEFAULTLABEL' | translate}}</p>
|
||||||
|
<div class="content" *ngIf="features">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<span class="left-desc">{{'FEATURES.DATA.AUDITLOGRETENTION' | translate}}</span>
|
<span class="left-desc">{{'FEATURES.DATA.AUDITLOGRETENTION' | translate}}</span>
|
||||||
<span class="fill-space"></span>
|
<span class="fill-space"></span>
|
||||||
<span>{{features.auditLogRetention | timestampToRetention }} {{'FEATURES.RETENTIONHOURS' | translate}}</span>
|
<span>{{features.auditLogRetention | timestampToRetention }} {{'FEATURES.RETENTIONHOURS' |
|
||||||
|
translate}}</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<span class="left-desc">{{'FEATURES.DATA.LOGINPOLICYUSERNAMELOGIN' | translate}}</span>
|
<span class="left-desc">{{'FEATURES.DATA.LOGINPOLICYUSERNAMELOGIN' | translate}}</span>
|
||||||
|
@ -1,8 +1,66 @@
|
|||||||
.default {
|
.default {
|
||||||
color: var(--color-main);
|
color: var(--color-main);
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tier-desc {
|
||||||
|
color: var(--grey);
|
||||||
|
font-size: 14px;
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.detail {
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 14px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-size: 14px;
|
||||||
|
color: var(--grey);
|
||||||
|
margin-bottom: .5rem;
|
||||||
|
|
||||||
|
a {
|
||||||
|
margin-left: .5rem;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
height: 15px;
|
||||||
|
width: auto;
|
||||||
|
margin-left: .5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner {
|
||||||
|
margin: .5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error {
|
||||||
|
color: #f44336;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.current-tier {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.divider {
|
||||||
|
height: 1px;
|
||||||
|
width: 100%;
|
||||||
|
background-color: var(--grey);
|
||||||
|
opacity: .5;
|
||||||
|
margin: .5rem 0;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
padding-top: 1rem;
|
padding-top: 1rem;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
import { Component, Injector, OnDestroy, Type } from '@angular/core';
|
import { Component, Injector, OnDestroy, Type } from '@angular/core';
|
||||||
|
import { MatDialog } from '@angular/material/dialog';
|
||||||
import { ActivatedRoute } from '@angular/router';
|
import { ActivatedRoute } from '@angular/router';
|
||||||
import { Subscription } from 'rxjs';
|
import { Subscription } from 'rxjs';
|
||||||
import { switchMap } from 'rxjs/operators';
|
import { switchMap } from 'rxjs/operators';
|
||||||
import {
|
import {
|
||||||
GetOrgFeaturesResponse,
|
GetOrgFeaturesResponse,
|
||||||
SetDefaultFeaturesRequest,
|
SetDefaultFeaturesRequest,
|
||||||
SetOrgFeaturesRequest,
|
SetOrgFeaturesRequest,
|
||||||
} from 'src/app/proto/generated/zitadel/admin_pb';
|
} from 'src/app/proto/generated/zitadel/admin_pb';
|
||||||
import { Features } from 'src/app/proto/generated/zitadel/features_pb';
|
import { Features } from 'src/app/proto/generated/zitadel/features_pb';
|
||||||
import { GetFeaturesResponse } from 'src/app/proto/generated/zitadel/management_pb';
|
import { GetFeaturesResponse } from 'src/app/proto/generated/zitadel/management_pb';
|
||||||
@ -13,132 +14,206 @@ import { Org } from 'src/app/proto/generated/zitadel/org_pb';
|
|||||||
import { AdminService } from 'src/app/services/admin.service';
|
import { AdminService } from 'src/app/services/admin.service';
|
||||||
import { ManagementService } from 'src/app/services/mgmt.service';
|
import { ManagementService } from 'src/app/services/mgmt.service';
|
||||||
import { StorageService } from 'src/app/services/storage.service';
|
import { StorageService } from 'src/app/services/storage.service';
|
||||||
|
import { StripeCustomer, SubscriptionService } from 'src/app/services/subscription.service';
|
||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
|
|
||||||
|
import { COUNTRIES, Country } from './country';
|
||||||
|
import { PaymentInfoDialogComponent } from './payment-info-dialog/payment-info-dialog.component';
|
||||||
|
|
||||||
export enum FeatureServiceType {
|
export enum FeatureServiceType {
|
||||||
MGMT,
|
MGMT,
|
||||||
ADMIN,
|
ADMIN,
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-features',
|
selector: 'app-features',
|
||||||
templateUrl: './features.component.html',
|
templateUrl: './features.component.html',
|
||||||
styleUrls: ['./features.component.scss'],
|
styleUrls: ['./features.component.scss'],
|
||||||
})
|
})
|
||||||
export class FeaturesComponent implements OnDestroy {
|
export class FeaturesComponent implements OnDestroy {
|
||||||
private managementService!: ManagementService;
|
private managementService!: ManagementService;
|
||||||
public serviceType!: FeatureServiceType;
|
public serviceType!: FeatureServiceType;
|
||||||
|
|
||||||
public features!: Features.AsObject;
|
public features!: Features.AsObject;
|
||||||
|
|
||||||
private sub: Subscription = new Subscription();
|
private sub: Subscription = new Subscription();
|
||||||
private org!: Org.AsObject;
|
public org!: Org.AsObject;
|
||||||
|
|
||||||
public FeatureServiceType: any = FeatureServiceType;
|
public FeatureServiceType: any = FeatureServiceType;
|
||||||
constructor(
|
|
||||||
private route: ActivatedRoute,
|
public customerLoading: boolean = false;
|
||||||
private toast: ToastService,
|
public stripeLoading: boolean = false;
|
||||||
private sessionStorage: StorageService,
|
public stripeURL: string = '';
|
||||||
private injector: Injector,
|
public stripeCustomer!: StripeCustomer;
|
||||||
private adminService: AdminService,
|
|
||||||
) {
|
constructor(
|
||||||
const temporg = this.sessionStorage.getItem('organization') as Org.AsObject;
|
private route: ActivatedRoute,
|
||||||
if (temporg) {
|
private toast: ToastService,
|
||||||
this.org = temporg;
|
private sessionStorage: StorageService,
|
||||||
}
|
private injector: Injector,
|
||||||
this.sub = this.route.data.pipe(switchMap(data => {
|
private adminService: AdminService,
|
||||||
this.serviceType = data.serviceType;
|
private subService: SubscriptionService,
|
||||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
private dialog: MatDialog,
|
||||||
this.managementService = this.injector.get(ManagementService as Type<ManagementService>);
|
) {
|
||||||
}
|
const temporg = this.sessionStorage.getItem('organization') as Org.AsObject;
|
||||||
return this.route.params;
|
if (temporg) {
|
||||||
})).subscribe(_ => {
|
this.org = temporg;
|
||||||
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.customerLoading = false;
|
||||||
|
this.stripeCustomer = payload;
|
||||||
|
if (this.customerValid) {
|
||||||
|
this.getLinkToStripe();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.customerLoading = false;
|
||||||
|
console.error(error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public ngOnDestroy(): void {
|
public ngOnDestroy(): void {
|
||||||
this.sub.unsubscribe();
|
this.sub.unsubscribe();
|
||||||
}
|
}
|
||||||
|
|
||||||
public fetchData(): void {
|
public updateCustomer(): void {
|
||||||
this.getData().then(resp => {
|
const dialogRefPhone = this.dialog.open(PaymentInfoDialogComponent, {
|
||||||
if (resp?.features) {
|
data: {
|
||||||
this.features = resp.features;
|
customer: this.stripeCustomer,
|
||||||
}
|
},
|
||||||
|
width: '400px',
|
||||||
|
});
|
||||||
|
|
||||||
|
dialogRefPhone.afterClosed().subscribe(customer => {
|
||||||
|
if (customer) {
|
||||||
|
console.log(customer);
|
||||||
|
this.stripeCustomer = customer;
|
||||||
|
this.subService.updateCustomer(this.org.id, customer).then(() => {
|
||||||
|
this.getLinkToStripe();
|
||||||
|
}).catch(console.error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public getLinkToStripe(): void {
|
||||||
|
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||||
|
this.stripeLoading = true;
|
||||||
|
this.subService.getLink(this.org.id, window.location.href)
|
||||||
|
.then(payload => {
|
||||||
|
this.stripeLoading = false;
|
||||||
|
console.log(payload);
|
||||||
|
this.stripeURL = payload.redirect_url;
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.stripeLoading = false;
|
||||||
|
console.error(error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async getData(): Promise<GetFeaturesResponse.AsObject | GetOrgFeaturesResponse.AsObject | undefined> {
|
public fetchData(): void {
|
||||||
switch (this.serviceType) {
|
this.getData().then(resp => {
|
||||||
case FeatureServiceType.MGMT:
|
if (resp?.features) {
|
||||||
return this.managementService.getFeatures();
|
this.features = resp.features;
|
||||||
case FeatureServiceType.ADMIN:
|
}
|
||||||
if (this.org?.id) {
|
});
|
||||||
return this.adminService.getDefaultFeatures();
|
}
|
||||||
}
|
|
||||||
break;
|
private async getData(): Promise<GetFeaturesResponse.AsObject | GetOrgFeaturesResponse.AsObject | undefined> {
|
||||||
|
switch (this.serviceType) {
|
||||||
|
case FeatureServiceType.MGMT:
|
||||||
|
return this.managementService.getFeatures();
|
||||||
|
case FeatureServiceType.ADMIN:
|
||||||
|
if (this.org?.id) {
|
||||||
|
return this.adminService.getDefaultFeatures();
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public savePolicy(): void {
|
public savePolicy(): void {
|
||||||
switch (this.serviceType) {
|
switch (this.serviceType) {
|
||||||
case FeatureServiceType.MGMT:
|
case FeatureServiceType.MGMT:
|
||||||
const req = new SetOrgFeaturesRequest();
|
const req = new SetOrgFeaturesRequest();
|
||||||
req.setOrgId(this.org.id);
|
req.setOrgId(this.org.id);
|
||||||
|
|
||||||
req.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
|
req.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
|
||||||
req.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
|
req.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
|
||||||
req.setLoginPolicyIdp(this.features.loginPolicyIdp);
|
req.setLoginPolicyIdp(this.features.loginPolicyIdp);
|
||||||
req.setLoginPolicyFactors(this.features.loginPolicyFactors);
|
req.setLoginPolicyFactors(this.features.loginPolicyFactors);
|
||||||
req.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
|
req.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
|
||||||
req.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
|
req.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
|
||||||
req.setLabelPolicy(this.features.labelPolicy);
|
req.setLabelPolicy(this.features.labelPolicy);
|
||||||
|
|
||||||
this.adminService.setOrgFeatures(req).then(() => {
|
this.adminService.setOrgFeatures(req).then(() => {
|
||||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||||
}).catch(error => {
|
}).catch(error => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
case FeatureServiceType.ADMIN:
|
case FeatureServiceType.ADMIN:
|
||||||
// update Default org iam policy?
|
// update Default org iam policy?
|
||||||
const dreq = new SetDefaultFeaturesRequest();
|
const dreq = new SetDefaultFeaturesRequest();
|
||||||
dreq.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
|
dreq.setLoginPolicyUsernameLogin(this.features.loginPolicyUsernameLogin);
|
||||||
dreq.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
|
dreq.setLoginPolicyRegistration(this.features.loginPolicyRegistration);
|
||||||
dreq.setLoginPolicyIdp(this.features.loginPolicyIdp);
|
dreq.setLoginPolicyIdp(this.features.loginPolicyIdp);
|
||||||
dreq.setLoginPolicyFactors(this.features.loginPolicyFactors);
|
dreq.setLoginPolicyFactors(this.features.loginPolicyFactors);
|
||||||
dreq.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
|
dreq.setLoginPolicyPasswordless(this.features.loginPolicyPasswordless);
|
||||||
dreq.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
|
dreq.setPasswordComplexityPolicy(this.features.passwordComplexityPolicy);
|
||||||
dreq.setLabelPolicy(this.features.labelPolicy);
|
dreq.setLabelPolicy(this.features.labelPolicy);
|
||||||
|
|
||||||
this.adminService.setDefaultFeatures(dreq).then(() => {
|
this.adminService.setDefaultFeatures(dreq).then(() => {
|
||||||
this.toast.showInfo('POLICY.TOAST.SET', true);
|
this.toast.showInfo('POLICY.TOAST.SET', true);
|
||||||
}).catch(error => {
|
}).catch(error => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public resetFeatures(): void {
|
public resetFeatures(): void {
|
||||||
if (this.serviceType === FeatureServiceType.MGMT) {
|
if (this.serviceType === FeatureServiceType.MGMT) {
|
||||||
this.adminService.resetOrgFeatures(this.org.id).then(() => {
|
this.adminService.resetOrgFeatures(this.org.id).then(() => {
|
||||||
this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true);
|
this.toast.showInfo('POLICY.TOAST.RESETSUCCESS', true);
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.fetchData();
|
this.fetchData();
|
||||||
}, 1000);
|
}, 1000);
|
||||||
}).catch(error => {
|
}).catch(error => {
|
||||||
this.toast.showError(error);
|
this.toast.showError(error);
|
||||||
});
|
});
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public get isDefault(): boolean {
|
public get isDefault(): boolean {
|
||||||
if (this.features && this.serviceType === FeatureServiceType.MGMT) {
|
if (this.features && this.serviceType === FeatureServiceType.MGMT) {
|
||||||
return this.features.isDefault;
|
return this.features.isDefault;
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get customerValid(): boolean {
|
||||||
|
return !!this.stripeCustomer?.contact &&
|
||||||
|
!!this.stripeCustomer?.address &&
|
||||||
|
!!this.stripeCustomer?.city &&
|
||||||
|
!!this.stripeCustomer?.postal_code;
|
||||||
|
}
|
||||||
|
|
||||||
|
get customerCountry(): Country | undefined {
|
||||||
|
return COUNTRIES.find(country => country.isoCode === this.stripeCustomer.country);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
import { CommonModule } from '@angular/common';
|
import { CommonModule } from '@angular/common';
|
||||||
import { NgModule } from '@angular/core';
|
import { NgModule } from '@angular/core';
|
||||||
import { FormsModule } from '@angular/forms';
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
||||||
import { MatButtonModule } from '@angular/material/button';
|
import { MatButtonModule } from '@angular/material/button';
|
||||||
import { MatIconModule } from '@angular/material/icon';
|
import { MatIconModule } from '@angular/material/icon';
|
||||||
|
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
|
||||||
|
import { MatSelectModule } from '@angular/material/select';
|
||||||
import { MatSlideToggleModule } from '@angular/material/slide-toggle';
|
import { MatSlideToggleModule } from '@angular/material/slide-toggle';
|
||||||
import { MatTooltipModule } from '@angular/material/tooltip';
|
import { MatTooltipModule } from '@angular/material/tooltip';
|
||||||
import { TranslateModule } from '@ngx-translate/core';
|
import { TranslateModule } from '@ngx-translate/core';
|
||||||
@ -11,29 +13,37 @@ import { DetailLayoutModule } from 'src/app/modules/detail-layout/detail-layout.
|
|||||||
import { InputModule } from 'src/app/modules/input/input.module';
|
import { InputModule } from 'src/app/modules/input/input.module';
|
||||||
import { HasRolePipeModule } from 'src/app/pipes/has-role-pipe/has-role-pipe.module';
|
import { HasRolePipeModule } from 'src/app/pipes/has-role-pipe/has-role-pipe.module';
|
||||||
import {
|
import {
|
||||||
TimestampToRetentionPipeModule,
|
TimestampToRetentionPipeModule,
|
||||||
} from 'src/app/pipes/timestamp-to-retention-pipe/timestamp-to-retention-pipe.module';
|
} from 'src/app/pipes/timestamp-to-retention-pipe/timestamp-to-retention-pipe.module';
|
||||||
|
|
||||||
|
import { FormFieldModule } from '../form-field/form-field.module';
|
||||||
import { InfoSectionModule } from '../info-section/info-section.module';
|
import { InfoSectionModule } from '../info-section/info-section.module';
|
||||||
import { FeaturesRoutingModule } from './features-routing.module';
|
import { FeaturesRoutingModule } from './features-routing.module';
|
||||||
import { FeaturesComponent } from './features.component';
|
import { FeaturesComponent } from './features.component';
|
||||||
|
import { PaymentInfoDialogComponent } from './payment-info-dialog/payment-info-dialog.component';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
FeaturesComponent,
|
FeaturesComponent,
|
||||||
|
PaymentInfoDialogComponent,
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
FeaturesRoutingModule,
|
FeaturesRoutingModule,
|
||||||
CommonModule,
|
CommonModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
InputModule,
|
InputModule,
|
||||||
MatButtonModule,
|
MatButtonModule,
|
||||||
|
FormFieldModule,
|
||||||
|
InputModule,
|
||||||
HasRoleModule,
|
HasRoleModule,
|
||||||
MatSlideToggleModule,
|
MatSlideToggleModule,
|
||||||
|
MatSelectModule,
|
||||||
MatIconModule,
|
MatIconModule,
|
||||||
HasRoleModule,
|
HasRoleModule,
|
||||||
HasRolePipeModule,
|
HasRolePipeModule,
|
||||||
MatTooltipModule,
|
MatTooltipModule,
|
||||||
|
MatProgressSpinnerModule,
|
||||||
InfoSectionModule,
|
InfoSectionModule,
|
||||||
TranslateModule,
|
TranslateModule,
|
||||||
DetailLayoutModule,
|
DetailLayoutModule,
|
||||||
|
@ -0,0 +1,60 @@
|
|||||||
|
<h1 mat-dialog-title>
|
||||||
|
<span class="title">{{'FEATURES.TIER.DETAILS' | translate}} {{data?.number}}</span>
|
||||||
|
</h1>
|
||||||
|
<p class="desc">{{'FEATURES.TIER.DETAILS_DESC' | translate}}</p>
|
||||||
|
<mat-spinner class="spinner" *ngIf="stripeLoading" diameter="20"></mat-spinner>
|
||||||
|
|
||||||
|
<div mat-dialog-content>
|
||||||
|
<form [formGroup]="form">
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.CONTACT' | translate }}</cnsl-label>
|
||||||
|
<input cnslInput name="contact" formControlName="contact" />
|
||||||
|
</cnsl-form-field>
|
||||||
|
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.COMPANY' | translate }}</cnsl-label>
|
||||||
|
<input cnslInput name="company" formControlName="company" />
|
||||||
|
</cnsl-form-field>
|
||||||
|
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.ADDRESS' | translate }}</cnsl-label>
|
||||||
|
<input cnslInput name="address" formControlName="address" />
|
||||||
|
</cnsl-form-field>
|
||||||
|
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.CITY' | translate }}</cnsl-label>
|
||||||
|
<input cnslInput name="city" formControlName="city" />
|
||||||
|
</cnsl-form-field>
|
||||||
|
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.POSTAL_CODE' | translate }}</cnsl-label>
|
||||||
|
<input cnslInput name="postal_code" formControlName="postal_code" />
|
||||||
|
</cnsl-form-field>
|
||||||
|
|
||||||
|
<cnsl-form-field class="formfield">
|
||||||
|
<cnsl-label>{{ 'FEATURES.TIER.COUNTRY' | translate }}</cnsl-label>
|
||||||
|
<mat-select formControlName="country" name="country" (selectionChange)="changeCountry($event)">
|
||||||
|
<mat-option *ngFor="let country of COUNTRIES" [value]="country.isoCode" class="center-row">
|
||||||
|
<div class="center">
|
||||||
|
<div class="img-wrapper">
|
||||||
|
<img height="20px" width="30px"
|
||||||
|
style="margin-right: 1rem; border-radius: 2px; vertical-align: middle;"
|
||||||
|
src="../../../../assets/flags/{{country.isoCode.toLowerCase()}}.png" />
|
||||||
|
</div>
|
||||||
|
{{country.isoCode}} <span class="lighter">| {{country.name}}</span>
|
||||||
|
</div>
|
||||||
|
</mat-option>
|
||||||
|
</mat-select>
|
||||||
|
</cnsl-form-field>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<div mat-dialog-actions class="action">
|
||||||
|
<button cdkFocusInitial color="primary" mat-button class="ok-button" (click)="dialogRef.close()">
|
||||||
|
{{'ACTIONS.CLOSE' | translate}}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button [disabled]="form.invalid" cdkFocusInitial color="primary" mat-raised-button class="ok-button"
|
||||||
|
(click)="submitAndCloseDialog()">
|
||||||
|
{{'ACTIONS.OK' | translate}}
|
||||||
|
</button>
|
||||||
|
</div>
|
@ -0,0 +1,47 @@
|
|||||||
|
h1 {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.desc {
|
||||||
|
font-size: 14px;
|
||||||
|
color: var(--grey);
|
||||||
|
}
|
||||||
|
|
||||||
|
.formfield {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
|
||||||
|
.ok-button {
|
||||||
|
margin-left: .5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
border-radius: .5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
height: 15px;
|
||||||
|
width: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-wrapper {
|
||||||
|
width: 50px;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.lighter {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--grey);
|
||||||
|
padding: 0 .5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.center {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
@ -0,0 +1,25 @@
|
|||||||
|
import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
|
||||||
|
|
||||||
|
import { PaymentInfoDialogComponent } from './payment-info-dialog.component';
|
||||||
|
|
||||||
|
describe('PaymentInfoDialogComponent', () => {
|
||||||
|
let component: PaymentInfoDialogComponent;
|
||||||
|
let fixture: ComponentFixture<PaymentInfoDialogComponent>;
|
||||||
|
|
||||||
|
beforeEach(waitForAsync(() => {
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [PaymentInfoDialogComponent],
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
}));
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(PaymentInfoDialogComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create', () => {
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,91 @@
|
|||||||
|
import { Component, Inject } from '@angular/core';
|
||||||
|
import { AbstractControl, FormBuilder, FormGroup, Validators } from '@angular/forms';
|
||||||
|
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
|
||||||
|
import { MatSelectChange } from '@angular/material/select';
|
||||||
|
import { SubscriptionService } from 'src/app/services/subscription.service';
|
||||||
|
|
||||||
|
import { COUNTRIES, Country } from '../country';
|
||||||
|
|
||||||
|
function compare(a: Country, b: Country): number {
|
||||||
|
if (a.isoCode < b.isoCode) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (a.isoCode > b.isoCode) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-payment-info-dialog',
|
||||||
|
templateUrl: './payment-info-dialog.component.html',
|
||||||
|
styleUrls: ['./payment-info-dialog.component.scss'],
|
||||||
|
})
|
||||||
|
export class PaymentInfoDialogComponent {
|
||||||
|
public stripeLoading: boolean = false;
|
||||||
|
public COUNTRIES: Country[] = COUNTRIES.sort(compare);
|
||||||
|
public form!: FormGroup;
|
||||||
|
|
||||||
|
private orgId: string = '';
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private subService: SubscriptionService,
|
||||||
|
private fb: FormBuilder,
|
||||||
|
public dialogRef: MatDialogRef<PaymentInfoDialogComponent>,
|
||||||
|
@Inject(MAT_DIALOG_DATA) public data: any,
|
||||||
|
) {
|
||||||
|
this.orgId = data.orgId;
|
||||||
|
|
||||||
|
this.form = this.fb.group({
|
||||||
|
contact: ['', [Validators.required]],
|
||||||
|
company: ['', []],
|
||||||
|
address: ['', [Validators.required]],
|
||||||
|
city: ['', [Validators.required]],
|
||||||
|
postal_code: ['', [Validators.required]],
|
||||||
|
country: ['', [Validators.required]],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (data.customer) {
|
||||||
|
this.form.patchValue(data.customer);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.customer?.country) {
|
||||||
|
this.form.get('country')?.setValue('CH');
|
||||||
|
}
|
||||||
|
|
||||||
|
this.getLink();
|
||||||
|
}
|
||||||
|
|
||||||
|
public getLink(): void {
|
||||||
|
if (this.orgId) {
|
||||||
|
this.stripeLoading = true;
|
||||||
|
this.subService.getLink(this.orgId, window.location.href)
|
||||||
|
.then(payload => {
|
||||||
|
this.stripeLoading = false;
|
||||||
|
console.log(payload);
|
||||||
|
if (payload.redirect_url) {
|
||||||
|
window.open(payload.redirect_url, '_blank');
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.stripeLoading = false;
|
||||||
|
console.error(error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public changeCountry(selection: MatSelectChange): void {
|
||||||
|
const country = COUNTRIES.find(c => c.isoCode === selection.value);
|
||||||
|
if (country && country.phoneCode !== undefined && this.phone && this.phone.value !== `+${country.phoneCode}`) {
|
||||||
|
this.phone.setValue(`+${country.phoneCode}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
submitAndCloseDialog(): void {
|
||||||
|
this.dialogRef.close(this.form.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
get phone(): AbstractControl | null {
|
||||||
|
return this.form.get('phone');
|
||||||
|
}
|
||||||
|
}
|
@ -39,6 +39,16 @@
|
|||||||
<td mat-cell *matCellDef="let org"> {{org.name}} </td>
|
<td mat-cell *matCellDef="let org"> {{org.name}} </td>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
|
||||||
|
<ng-container matColumnDef="creationDate">
|
||||||
|
<th mat-header-cell *matHeaderCellDef mat-sort-header
|
||||||
|
[ngClass]="{'search-active': this.orgSearchKey == OrgListSearchKey.NAME}">
|
||||||
|
{{ 'ORG.PAGES.CREATIONDATE' | translate }}
|
||||||
|
</th>
|
||||||
|
<td mat-cell *matCellDef="let org">
|
||||||
|
{{org.details?.creationDate | timestampToDate | localizedDate: 'EEE dd. MMM YYYY, HH:mm'}}
|
||||||
|
</td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
||||||
<tr (click)="setAndNavigateToOrg(row)" mat-row *matRowDef="let row; columns: displayedColumns;"></tr>
|
<tr (click)="setAndNavigateToOrg(row)" mat-row *matRowDef="let row; columns: displayedColumns;"></tr>
|
||||||
</table>
|
</table>
|
||||||
|
@ -11,102 +11,102 @@ import { Org, OrgNameQuery, OrgQuery } from 'src/app/proto/generated/zitadel/org
|
|||||||
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
import { GrpcAuthService } from 'src/app/services/grpc-auth.service';
|
||||||
|
|
||||||
enum OrgListSearchKey {
|
enum OrgListSearchKey {
|
||||||
NAME = 'NAME',
|
NAME = 'NAME',
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-org-list',
|
selector: 'app-org-list',
|
||||||
templateUrl: './org-list.component.html',
|
templateUrl: './org-list.component.html',
|
||||||
styleUrls: ['./org-list.component.scss'],
|
styleUrls: ['./org-list.component.scss'],
|
||||||
animations: [
|
animations: [
|
||||||
enterAnimations,
|
enterAnimations,
|
||||||
],
|
],
|
||||||
})
|
})
|
||||||
export class OrgListComponent implements AfterViewInit {
|
export class OrgListComponent implements AfterViewInit {
|
||||||
public orgSearchKey: OrgListSearchKey | undefined = undefined;
|
public orgSearchKey: OrgListSearchKey | undefined = undefined;
|
||||||
|
|
||||||
@ViewChild(MatPaginator) public paginator!: MatPaginator;
|
@ViewChild(MatPaginator) public paginator!: MatPaginator;
|
||||||
@ViewChild(MatSort) sort!: MatSort;
|
@ViewChild(MatSort) sort!: MatSort;
|
||||||
@ViewChild('input') public filter!: Input;
|
@ViewChild('input') public filter!: Input;
|
||||||
|
|
||||||
public dataSource!: MatTableDataSource<Org.AsObject>;
|
public dataSource!: MatTableDataSource<Org.AsObject>;
|
||||||
public displayedColumns: string[] = ['select', 'id', 'name'];
|
public displayedColumns: string[] = ['select', 'id', 'name', 'creationDate'];
|
||||||
private loadingSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
private loadingSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
||||||
public loading$: Observable<boolean> = this.loadingSubject.asObservable();
|
public loading$: Observable<boolean> = this.loadingSubject.asObservable();
|
||||||
public activeOrg!: Org.AsObject;
|
public activeOrg!: Org.AsObject;
|
||||||
public OrgListSearchKey: any = OrgListSearchKey;
|
public OrgListSearchKey: any = OrgListSearchKey;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private authService: GrpcAuthService,
|
private authService: GrpcAuthService,
|
||||||
private router: Router,
|
private router: Router,
|
||||||
) {
|
) {
|
||||||
this.loadOrgs(10, 0);
|
this.loadOrgs(10, 0);
|
||||||
|
|
||||||
this.authService.getActiveOrg().then(org => this.activeOrg = org);
|
this.authService.getActiveOrg().then(org => this.activeOrg = org);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ngAfterViewInit(): void {
|
||||||
|
this.loadOrgs(10, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public loadOrgs(limit: number, offset: number, filter?: string): void {
|
||||||
|
this.loadingSubject.next(true);
|
||||||
|
let query;
|
||||||
|
if (filter) {
|
||||||
|
query = new OrgQuery();
|
||||||
|
const orgNameQuery = new OrgNameQuery();
|
||||||
|
orgNameQuery.setMethod(TextQueryMethod.TEXT_QUERY_METHOD_CONTAINS_IGNORE_CASE);
|
||||||
|
orgNameQuery.setName(filter);
|
||||||
|
query.setNameQuery(orgNameQuery);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ngAfterViewInit(): void {
|
from(this.authService.listMyProjectOrgs(limit, offset, query ? [query] : undefined)).pipe(
|
||||||
this.loadOrgs(10, 0);
|
map(resp => {
|
||||||
|
return resp.resultList;
|
||||||
|
}),
|
||||||
|
catchError(() => of([])),
|
||||||
|
finalize(() => this.loadingSubject.next(false)),
|
||||||
|
).subscribe(views => {
|
||||||
|
this.dataSource = new MatTableDataSource(views);
|
||||||
|
this.dataSource.paginator = this.paginator;
|
||||||
|
this.dataSource.sort = this.sort;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public selectOrg(item: Org.AsObject, event?: any): void {
|
||||||
|
this.authService.setActiveOrg(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
public refresh(): void {
|
||||||
|
this.loadOrgs(this.paginator.length, this.paginator.pageSize * this.paginator.pageIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
public setFilter(key: OrgListSearchKey): void {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (this.filter) {
|
||||||
|
(this.filter as any).nativeElement.focus();
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
|
||||||
|
if (this.orgSearchKey !== key) {
|
||||||
|
this.orgSearchKey = key;
|
||||||
|
} else {
|
||||||
|
this.orgSearchKey = undefined;
|
||||||
|
this.refresh();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public loadOrgs(limit: number, offset: number, filter?: string): void {
|
public applyFilter(event: Event): void {
|
||||||
this.loadingSubject.next(true);
|
const filterValue = (event.target as HTMLInputElement).value;
|
||||||
let query;
|
this.loadOrgs(
|
||||||
if (filter) {
|
this.paginator.pageSize,
|
||||||
query = new OrgQuery();
|
this.paginator.pageIndex * this.paginator.pageSize,
|
||||||
const orgNameQuery = new OrgNameQuery();
|
filterValue.trim().toLowerCase(),
|
||||||
orgNameQuery.setMethod(TextQueryMethod.TEXT_QUERY_METHOD_CONTAINS_IGNORE_CASE);
|
);
|
||||||
orgNameQuery.setName(filter);
|
}
|
||||||
query.setNameQuery(orgNameQuery);
|
|
||||||
}
|
|
||||||
|
|
||||||
from(this.authService.listMyProjectOrgs(limit, offset, query ? [query] : undefined)).pipe(
|
public setAndNavigateToOrg(org: Org.AsObject): void {
|
||||||
map(resp => {
|
this.authService.setActiveOrg(org);
|
||||||
return resp.resultList;
|
this.router.navigate(['/org']);
|
||||||
}),
|
}
|
||||||
catchError(() => of([])),
|
|
||||||
finalize(() => this.loadingSubject.next(false)),
|
|
||||||
).subscribe(views => {
|
|
||||||
this.dataSource = new MatTableDataSource(views);
|
|
||||||
this.dataSource.paginator = this.paginator;
|
|
||||||
this.dataSource.sort = this.sort;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public selectOrg(item: Org.AsObject, event?: any): void {
|
|
||||||
this.authService.setActiveOrg(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
public refresh(): void {
|
|
||||||
this.loadOrgs(this.paginator.length, this.paginator.pageSize * this.paginator.pageIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
public setFilter(key: OrgListSearchKey): void {
|
|
||||||
setTimeout(() => {
|
|
||||||
if (this.filter) {
|
|
||||||
(this.filter as any).nativeElement.focus();
|
|
||||||
}
|
|
||||||
}, 100);
|
|
||||||
|
|
||||||
if (this.orgSearchKey !== key) {
|
|
||||||
this.orgSearchKey = key;
|
|
||||||
} else {
|
|
||||||
this.orgSearchKey = undefined;
|
|
||||||
this.refresh();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public applyFilter(event: Event): void {
|
|
||||||
const filterValue = (event.target as HTMLInputElement).value;
|
|
||||||
this.loadOrgs(
|
|
||||||
this.paginator.pageSize,
|
|
||||||
this.paginator.pageIndex * this.paginator.pageSize,
|
|
||||||
filterValue.trim().toLowerCase(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
public setAndNavigateToOrg(org: Org.AsObject): void {
|
|
||||||
this.authService.setActiveOrg(org);
|
|
||||||
this.router.navigate(['/org']);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
89
console/src/app/services/subscription.service.ts
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
import { HttpClient } from '@angular/common/http';
|
||||||
|
import { Injectable } from '@angular/core';
|
||||||
|
|
||||||
|
import { StorageService } from './storage.service';
|
||||||
|
|
||||||
|
const authorizationKey = 'Authorization';
|
||||||
|
const bearerPrefix = 'Bearer';
|
||||||
|
const accessTokenStorageKey = 'access_token';
|
||||||
|
|
||||||
|
export interface StripeCustomer {
|
||||||
|
contact: string;
|
||||||
|
company?: string;
|
||||||
|
address: string;
|
||||||
|
city: string;
|
||||||
|
postal_code: string;
|
||||||
|
country: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Injectable({
|
||||||
|
providedIn: 'root',
|
||||||
|
})
|
||||||
|
export class SubscriptionService {
|
||||||
|
constructor(private http: HttpClient, private storageService: StorageService) { }
|
||||||
|
|
||||||
|
public getLink(orgId: string, redirectURI: string): Promise<any> {
|
||||||
|
return this.http.get('./assets/environment.json')
|
||||||
|
.toPromise().then((data: any) => {
|
||||||
|
if (data && data.subscriptionServiceUrl) {
|
||||||
|
const serviceUrl = data.subscriptionServiceUrl;
|
||||||
|
const accessToken = this.storageService.getItem(accessTokenStorageKey);
|
||||||
|
return this.http.get(`${serviceUrl}/redirect`, {
|
||||||
|
headers: {
|
||||||
|
// 'Content-Type': 'application/json; charset=utf-8',
|
||||||
|
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
'org': orgId,
|
||||||
|
'return_url': encodeURI(redirectURI),
|
||||||
|
'country': 'ch',
|
||||||
|
},
|
||||||
|
}).toPromise();
|
||||||
|
} else {
|
||||||
|
return Promise.reject('Could not load environment');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public getCustomer(orgId: string): Promise<any> {
|
||||||
|
return this.http.get('./assets/environment.json')
|
||||||
|
.toPromise().then((data: any) => {
|
||||||
|
if (data && data.subscriptionServiceUrl) {
|
||||||
|
const serviceUrl = data.subscriptionServiceUrl;
|
||||||
|
const accessToken = this.storageService.getItem(accessTokenStorageKey);
|
||||||
|
return this.http.get(`${serviceUrl}/customer`, {
|
||||||
|
headers: {
|
||||||
|
// 'Content-Type': 'application/json; charset=utf-8',
|
||||||
|
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
'org': orgId,
|
||||||
|
},
|
||||||
|
}).toPromise();
|
||||||
|
} else {
|
||||||
|
return Promise.reject('Could not load environment');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public updateCustomer(orgId: string, body: StripeCustomer): Promise<any> {
|
||||||
|
return this.http.get('./assets/environment.json')
|
||||||
|
.toPromise().then((data: any) => {
|
||||||
|
if (data && data.subscriptionServiceUrl) {
|
||||||
|
const serviceUrl = data.subscriptionServiceUrl;
|
||||||
|
const accessToken = this.storageService.getItem(accessTokenStorageKey);
|
||||||
|
return this.http.put(`${serviceUrl}/customer`, body, {
|
||||||
|
headers: {
|
||||||
|
// 'Content-Type': 'application/json; charset=utf-8',
|
||||||
|
[authorizationKey]: `${bearerPrefix} ${accessToken}`,
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
'org': orgId,
|
||||||
|
},
|
||||||
|
}).toPromise();
|
||||||
|
} else {
|
||||||
|
return Promise.reject('Could not load environment');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -2,6 +2,7 @@
|
|||||||
"authServiceUrl": "https://api.zitadel.io",
|
"authServiceUrl": "https://api.zitadel.io",
|
||||||
"mgmtServiceUrl": "https://api.zitadel.io",
|
"mgmtServiceUrl": "https://api.zitadel.io",
|
||||||
"adminServiceUrl":"https://api.zitadel.io",
|
"adminServiceUrl":"https://api.zitadel.io",
|
||||||
|
"subscriptionServiceUrl":"https://sub.zitadel.io",
|
||||||
"issuer": "https://issuer.zitadel.io",
|
"issuer": "https://issuer.zitadel.io",
|
||||||
"clientid": "69234247558357051@zitadel"
|
"clientid": "69234247558357051@zitadel"
|
||||||
}
|
}
|
||||||
|
BIN
console/src/assets/flags/ad.png
Normal file
After Width: | Height: | Size: 652 B |
BIN
console/src/assets/flags/ae.png
Normal file
After Width: | Height: | Size: 170 B |
BIN
console/src/assets/flags/af.png
Normal file
After Width: | Height: | Size: 620 B |
BIN
console/src/assets/flags/ag.png
Normal file
After Width: | Height: | Size: 735 B |
BIN
console/src/assets/flags/ai.png
Normal file
After Width: | Height: | Size: 646 B |
BIN
console/src/assets/flags/al.png
Normal file
After Width: | Height: | Size: 688 B |
BIN
console/src/assets/flags/am.png
Normal file
After Width: | Height: | Size: 122 B |
BIN
console/src/assets/flags/an.png
Normal file
After Width: | Height: | Size: 309 B |
BIN
console/src/assets/flags/ao.png
Normal file
After Width: | Height: | Size: 549 B |
BIN
console/src/assets/flags/aq.png
Normal file
After Width: | Height: | Size: 1.9 KiB |
BIN
console/src/assets/flags/ar.png
Normal file
After Width: | Height: | Size: 327 B |
BIN
console/src/assets/flags/as.png
Normal file
After Width: | Height: | Size: 759 B |
BIN
console/src/assets/flags/at.png
Normal file
After Width: | Height: | Size: 133 B |
BIN
console/src/assets/flags/au.png
Normal file
After Width: | Height: | Size: 630 B |
BIN
console/src/assets/flags/aw.png
Normal file
After Width: | Height: | Size: 281 B |
BIN
console/src/assets/flags/ax.png
Normal file
After Width: | Height: | Size: 279 B |
BIN
console/src/assets/flags/az.png
Normal file
After Width: | Height: | Size: 243 B |
BIN
console/src/assets/flags/ba.png
Normal file
After Width: | Height: | Size: 461 B |
BIN
console/src/assets/flags/bb.png
Normal file
After Width: | Height: | Size: 381 B |
BIN
console/src/assets/flags/bd.png
Normal file
After Width: | Height: | Size: 258 B |
BIN
console/src/assets/flags/be.png
Normal file
After Width: | Height: | Size: 172 B |
BIN
console/src/assets/flags/bf.png
Normal file
After Width: | Height: | Size: 283 B |
BIN
console/src/assets/flags/bg.png
Normal file
After Width: | Height: | Size: 106 B |
BIN
console/src/assets/flags/bh.png
Normal file
After Width: | Height: | Size: 188 B |
BIN
console/src/assets/flags/bi.png
Normal file
After Width: | Height: | Size: 691 B |
BIN
console/src/assets/flags/bj.png
Normal file
After Width: | Height: | Size: 169 B |
BIN
console/src/assets/flags/bl.png
Normal file
After Width: | Height: | Size: 2.2 KiB |
BIN
console/src/assets/flags/bm.png
Normal file
After Width: | Height: | Size: 918 B |
BIN
console/src/assets/flags/bn.png
Normal file
After Width: | Height: | Size: 955 B |
BIN
console/src/assets/flags/bo.png
Normal file
After Width: | Height: | Size: 547 B |
BIN
console/src/assets/flags/bq.png
Normal file
After Width: | Height: | Size: 159 B |
BIN
console/src/assets/flags/br.png
Normal file
After Width: | Height: | Size: 750 B |
BIN
console/src/assets/flags/bs.png
Normal file
After Width: | Height: | Size: 289 B |
BIN
console/src/assets/flags/bt.png
Normal file
After Width: | Height: | Size: 1.3 KiB |
BIN
console/src/assets/flags/bv.png
Normal file
After Width: | Height: | Size: 260 B |
BIN
console/src/assets/flags/bw.png
Normal file
After Width: | Height: | Size: 172 B |
BIN
console/src/assets/flags/by.png
Normal file
After Width: | Height: | Size: 452 B |
BIN
console/src/assets/flags/bz.png
Normal file
After Width: | Height: | Size: 2.2 KiB |
BIN
console/src/assets/flags/ca.png
Normal file
After Width: | Height: | Size: 406 B |
BIN
console/src/assets/flags/cc.png
Normal file
After Width: | Height: | Size: 593 B |
BIN
console/src/assets/flags/cd.png
Normal file
After Width: | Height: | Size: 449 B |
BIN
console/src/assets/flags/cf.png
Normal file
After Width: | Height: | Size: 327 B |
BIN
console/src/assets/flags/cg.png
Normal file
After Width: | Height: | Size: 296 B |
BIN
console/src/assets/flags/ch.png
Normal file
After Width: | Height: | Size: 172 B |
BIN
console/src/assets/flags/ci.png
Normal file
After Width: | Height: | Size: 165 B |
BIN
console/src/assets/flags/ck.png
Normal file
After Width: | Height: | Size: 722 B |
BIN
console/src/assets/flags/cl.png
Normal file
After Width: | Height: | Size: 285 B |
BIN
console/src/assets/flags/cm.png
Normal file
After Width: | Height: | Size: 245 B |
BIN
console/src/assets/flags/cn.png
Normal file
After Width: | Height: | Size: 315 B |
BIN
console/src/assets/flags/co.png
Normal file
After Width: | Height: | Size: 158 B |
BIN
console/src/assets/flags/cr.png
Normal file
After Width: | Height: | Size: 109 B |
BIN
console/src/assets/flags/cu.png
Normal file
After Width: | Height: | Size: 356 B |
BIN
console/src/assets/flags/cv.png
Normal file
After Width: | Height: | Size: 407 B |
BIN
console/src/assets/flags/cw.png
Normal file
After Width: | Height: | Size: 296 B |
BIN
console/src/assets/flags/cx.png
Normal file
After Width: | Height: | Size: 720 B |
BIN
console/src/assets/flags/cy.png
Normal file
After Width: | Height: | Size: 572 B |
BIN
console/src/assets/flags/cz.png
Normal file
After Width: | Height: | Size: 341 B |
BIN
console/src/assets/flags/de.png
Normal file
After Width: | Height: | Size: 106 B |
BIN
console/src/assets/flags/dj.png
Normal file
After Width: | Height: | Size: 531 B |
BIN
console/src/assets/flags/dk.png
Normal file
After Width: | Height: | Size: 203 B |
BIN
console/src/assets/flags/dm.png
Normal file
After Width: | Height: | Size: 515 B |
BIN
console/src/assets/flags/do.png
Normal file
After Width: | Height: | Size: 422 B |
BIN
console/src/assets/flags/dz.png
Normal file
After Width: | Height: | Size: 405 B |
BIN
console/src/assets/flags/ec.png
Normal file
After Width: | Height: | Size: 1.0 KiB |
BIN
console/src/assets/flags/ee.png
Normal file
After Width: | Height: | Size: 158 B |
BIN
console/src/assets/flags/eg.png
Normal file
After Width: | Height: | Size: 366 B |
BIN
console/src/assets/flags/eh.png
Normal file
After Width: | Height: | Size: 374 B |
BIN
console/src/assets/flags/er.png
Normal file
After Width: | Height: | Size: 584 B |
BIN
console/src/assets/flags/es.png
Normal file
After Width: | Height: | Size: 682 B |
BIN
console/src/assets/flags/et.png
Normal file
After Width: | Height: | Size: 596 B |
BIN
console/src/assets/flags/eu.png
Normal file
After Width: | Height: | Size: 546 B |
BIN
console/src/assets/flags/fi.png
Normal file
After Width: | Height: | Size: 186 B |
BIN
console/src/assets/flags/fj.png
Normal file
After Width: | Height: | Size: 876 B |
BIN
console/src/assets/flags/fk.png
Normal file
After Width: | Height: | Size: 1.0 KiB |
BIN
console/src/assets/flags/fm.png
Normal file
After Width: | Height: | Size: 269 B |
BIN
console/src/assets/flags/fo.png
Normal file
After Width: | Height: | Size: 260 B |
BIN
console/src/assets/flags/fr.png
Normal file
After Width: | Height: | Size: 165 B |
BIN
console/src/assets/flags/ga.png
Normal file
After Width: | Height: | Size: 109 B |
BIN
console/src/assets/flags/gb-eng.png
Normal file
After Width: | Height: | Size: 99 B |
BIN
console/src/assets/flags/gb-nir.png
Normal file
After Width: | Height: | Size: 384 B |
BIN
console/src/assets/flags/gb-sct.png
Normal file
After Width: | Height: | Size: 358 B |
BIN
console/src/assets/flags/gb-wls.png
Normal file
After Width: | Height: | Size: 1.7 KiB |
BIN
console/src/assets/flags/gb.png
Normal file
After Width: | Height: | Size: 383 B |
BIN
console/src/assets/flags/gd.png
Normal file
After Width: | Height: | Size: 594 B |
BIN
console/src/assets/flags/ge.png
Normal file
After Width: | Height: | Size: 359 B |
BIN
console/src/assets/flags/gf.png
Normal file
After Width: | Height: | Size: 424 B |