fix(console): granted project member actions (create, update, delete) (#438)

* project grant members abst

* ref name from both project types
This commit is contained in:
Max Peintner 2020-07-09 18:02:05 +02:00 committed by GitHub
parent da113ffb95
commit 4fa68ae2ee
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 159 additions and 125 deletions

View File

@ -52,67 +52,68 @@
[opened]="!(isHandset$ | async)"> [opened]="!(isHandset$ | async)">
<div class="side-column"> <div class="side-column">
<div class="list"> <div class="list">
<a *ngIf="authService.authenticationChanged | async" class="nav-item" <ng-container *ngIf="authService.authenticationChanged | async">
[routerLinkActive]="['active']" [routerLinkActiveOptions]="{ exact: true }" <a class="nav-item" [routerLinkActive]="['active']" [routerLinkActiveOptions]="{ exact: true }"
[routerLink]="['/user/me']"> [routerLink]="['/user/me']">
<i class="icon las la-user-circle"></i> <i class="icon las la-user-circle"></i>
<span class="label">{{ 'MENU.PERSONAL_INFO' | translate }}</span> <span class="label">{{ 'MENU.PERSONAL_INFO' | translate }}</span>
</a> </a>
</ng-container>
<div *ngIf="authService.authenticationChanged | async" class="divider"> <ng-container *ngIf="iamreadwrite">
<div class="line"></div> <div class="divider">
</div> <div class="line"></div>
</div>
<a class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/iam']">
<i class="icon las la-gem"></i>
<span class="label">{{'MENU.IAM' | translate}}</span>
</a>
</ng-container>
<a *ngIf="iamreadwrite" class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/iam']"> <ng-container *ngIf="showOrgSection">
<i class="icon las la-gem"></i> <a class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/org']">
<span class="label">{{'MENU.IAM' | translate}}</span> <i class="icon las la-archway"></i>
</a> <span class="label">{{org?.name ? org.name : 'MENU.ORGANIZATION' | translate}}</span>
</a>
</ng-container>
<a *ngIf="showOrgSection" class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/org']"> <ng-container *ngIf="showProjectSection">
<i class="icon las la-archway"></i> <div class="divider">
<span class="label">{{org?.name ? org.name : 'MENU.ORGANIZATION' | translate}}</span> <div class="line"></div>
</a> <span>{{'MENU.PROJECTSSECTION' | translate}}</span>
<div class="line"></div>
</div>
<div *ngIf="showOrgSection" class="divider"> <a class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/projects']">
<div class="line"></div> <i class="icon las la-layer-group"></i>
<span>{{'MENU.PROJECTSSECTION' | translate}}</span> <span class="label">{{org?.name ? org.name : 'MENU.ORGANIZATION' | translate}}
<div class="line"></div> {{ 'MENU.PROJECT' | translate }}</span>
</div> </a>
<a *ngIf="showProjectSection" class="nav-item" [routerLinkActive]="['active']" <a class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/granted-projects']">
[routerLink]="[ '/projects']"> <i class="icon las la-layer-group"></i>
<i class="icon las la-layer-group"></i> <span class="label">{{ 'MENU.GRANTEDPROJECT' | translate }}</span>
<span class="label">{{org?.name ? org.name : 'MENU.ORGANIZATION' | translate}} </a>
{{ 'MENU.PROJECT' | translate }}</span> </ng-container>
</a>
<a *ngIf="showProjectSection" class="nav-item" [routerLinkActive]="['active']" <ng-container *ngIf="showUserSection">
[routerLink]="[ '/granted-projects']"> <div class="divider">
<i class="icon las la-layer-group"></i> <div class="line"></div>
<span class="label">{{ 'MENU.GRANTEDPROJECT' | translate }}</span> <span class="label">
</a> {{ 'MENU.USERSECTION' | translate }}</span>
<div class="line"></div>
</div>
<div *ngIf="showProjectSection" class="divider"> <a class="nav-item" [routerLinkActive]="['active']" [routerLink]="[ '/users']"
<div class="line"></div> [routerLinkActiveOptions]="{ exact: true }">
<span class="label"> <i class="icon las la-users"></i>
{{ 'MENU.USERSECTION' | translate }}</span> <span class="label">{{ 'MENU.USER' | translate }}</span>
<div class="line"></div> </a>
</ng-container>
</div>
<a *ngIf="showUserSection" class="nav-item" [routerLinkActive]="['active']"
[routerLink]="[ '/users']" [routerLinkActiveOptions]="{ exact: true }">
<i class="icon las la-users"></i>
<span class="label">{{ 'MENU.USER' | translate }}</span>
</a>
<span class="fill-space"></span> <span class="fill-space"></span>
</div> </div>
<span class="fill-space"></span> <span class="fill-space"></span>
<!-- <div class="footer">
<a href="https://caos.ch/impressum/" target="_blank" rel="noreferrer">AGB</a>
<a href="https://caos.ch/impressum/" target="_blank" rel="noreferrer">Impressum</a>
</div> -->
</div> </div>
</mat-drawer> </mat-drawer>
<mat-drawer-content class="content"> <mat-drawer-content class="content">

View File

@ -33,6 +33,8 @@ export class MemberCreateDialogComponent {
this.creationType = data.creationType; this.creationType = data.creationType;
this.projectId = data.projectId; this.projectId = data.projectId;
console.log(this.creationType);
if (this.creationType === CreationType.PROJECT_GRANTED) { if (this.creationType === CreationType.PROJECT_GRANTED) {
this.projectService.GetProjectGrantMemberRoles().then(resp => { this.projectService.GetProjectGrantMemberRoles().then(resp => {
this.memberRoleOptions = resp.toObject().rolesList; this.memberRoleOptions = resp.toObject().rolesList;

View File

@ -24,6 +24,7 @@ import { CreationType, MemberCreateDialogComponent } from '../../modules/add-mem
}) })
export class ProjectContributorsComponent implements OnInit { export class ProjectContributorsComponent implements OnInit {
@Input() public project!: ProjectView.AsObject | ProjectGrantView.AsObject; @Input() public project!: ProjectView.AsObject | ProjectGrantView.AsObject;
@Input() public grantId: string = '';
@Input() public projectType!: ProjectType; @Input() public projectType!: ProjectType;
@Input() public disabled: boolean = false; @Input() public disabled: boolean = false;
@ -65,7 +66,9 @@ export class ProjectContributorsComponent implements OnInit {
public openAddMember(): void { public openAddMember(): void {
const dialogRef = this.dialog.open(MemberCreateDialogComponent, { const dialogRef = this.dialog.open(MemberCreateDialogComponent, {
data: { data: {
creationType: CreationType.PROJECT_OWNED, // TODO replace
creationType: this.projectType === ProjectType.PROJECTTYPE_OWNED ? CreationType.PROJECT_OWNED :
ProjectType.PROJECTTYPE_GRANTED ? CreationType.PROJECT_GRANTED : ProjectType.PROJECTTYPE_OWNED,
projectId: this.project.projectId, projectId: this.project.projectId,
}, },
width: '400px', width: '400px',
@ -77,12 +80,27 @@ export class ProjectContributorsComponent implements OnInit {
const roles: string[] = resp.roles; const roles: string[] = resp.roles;
if (users && users.length && roles && roles.length) { if (users && users.length && roles && roles.length) {
Promise.all(users.map(user => { users.forEach(user => {
return this.projectService.AddProjectMember(this.project.projectId, user.id, roles); switch (this.projectType) {
})).then(() => { case ProjectType.PROJECTTYPE_OWNED:
this.toast.showError('members added'); return this.projectService.AddProjectMember(this.project.projectId, user.id, roles)
}).catch(error => { .then(() => {
this.toast.showError(error); this.toast.showInfo('members added');
}).catch(error => {
this.toast.showError(error);
});
case ProjectType.PROJECTTYPE_GRANTED:
return this.projectService.AddProjectGrantMember(
this.project.projectId,
this.grantId,
user.id,
roles,
).then(() => {
this.toast.showInfo('members added');
}).catch(error => {
this.toast.showError(error);
});
}
}); });
} }
} }
@ -92,7 +110,7 @@ export class ProjectContributorsComponent implements OnInit {
public showDetail(): void { public showDetail(): void {
if (this.project?.state === ProjectState.PROJECTSTATE_ACTIVE) { if (this.project?.state === ProjectState.PROJECTSTATE_ACTIVE) {
if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) { if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) {
this.router.navigate(['granted-projects', this.project.projectId, 'members']); this.router.navigate(['granted-projects', this.project.projectId, 'grant', this.grantId, 'members']);
} else if (this.projectType === ProjectType.PROJECTTYPE_OWNED) { } else if (this.projectType === ProjectType.PROJECTTYPE_OWNED) {
this.router.navigate(['projects', this.project.projectId, 'members']); this.router.navigate(['projects', this.project.projectId, 'members']);
} }

View File

@ -1,7 +1,7 @@
import { DataSource } from '@angular/cdk/collections'; import { DataSource } from '@angular/cdk/collections';
import { BehaviorSubject, from, Observable, of } from 'rxjs'; import { BehaviorSubject, from, Observable, of } from 'rxjs';
import { catchError, finalize, map } from 'rxjs/operators'; import { catchError, finalize, map } from 'rxjs/operators';
import { ProjectMember, ProjectMemberSearchResponse, ProjectType, ProjectView } from 'src/app/proto/generated/management_pb'; import { ProjectMember, ProjectMemberSearchResponse, ProjectType } from 'src/app/proto/generated/management_pb';
import { ProjectService } from 'src/app/services/project.service'; import { ProjectService } from 'src/app/services/project.service';
/** /**
@ -19,18 +19,19 @@ export class ProjectMembersDataSource extends DataSource<ProjectMember.AsObject>
super(); super();
} }
public loadMembers(project: ProjectView.AsObject, public loadMembers(projectId: string,
projectType: ProjectType, projectType: ProjectType,
pageIndex: number, pageSize: number, grantId?: string, sortDirection?: string): void { pageIndex: number, pageSize: number, grantId?: string): void {
const offset = pageIndex * pageSize; const offset = pageIndex * pageSize;
this.loadingSubject.next(true); this.loadingSubject.next(true);
console.log(grantId);
// TODO // TODO
const promise: Promise<ProjectMemberSearchResponse> | undefined = const promise: Promise<ProjectMemberSearchResponse> | undefined =
projectType === ProjectType.PROJECTTYPE_OWNED ? projectType === ProjectType.PROJECTTYPE_OWNED ?
this.projectService.SearchProjectMembers(project.projectId, pageSize, offset) : this.projectService.SearchProjectMembers(projectId, pageSize, offset) :
projectType === ProjectType.PROJECTTYPE_GRANTED && grantId ? projectType === ProjectType.PROJECTTYPE_GRANTED && grantId ?
this.projectService.SearchProjectGrantMembers(project.projectId, this.projectService.SearchProjectGrantMembers(projectId,
grantId, pageSize, offset) : undefined; grantId, pageSize, offset) : undefined;
if (promise) { if (promise) {
from(promise).pipe( from(promise).pipe(

View File

@ -7,7 +7,7 @@
</div> </div>
<div class="right"> <div class="right">
<div class="head"> <div class="head">
<h1>{{project?.name}} {{ 'PROJECT.MEMBER.TITLE' | translate }}</h1> <h1>{{projectName}} {{ 'PROJECT.MEMBER.TITLE' | translate }}</h1>
<p class="desc">{{ 'PROJECT.MEMBER.DESCRIPTION' | translate }}</p> <p class="desc">{{ 'PROJECT.MEMBER.DESCRIPTION' | translate }}</p>
</div> </div>

View File

@ -1,12 +1,12 @@
import { SelectionModel } from '@angular/cdk/collections'; import { SelectionModel } from '@angular/cdk/collections';
import { AfterViewInit, Component, ViewChild } from '@angular/core'; import { Component, ViewChild } from '@angular/core';
import { MatDialog } from '@angular/material/dialog'; import { MatDialog } from '@angular/material/dialog';
import { MatPaginator } from '@angular/material/paginator'; import { MatPaginator } from '@angular/material/paginator';
import { MatSelectChange } from '@angular/material/select'; import { MatSelectChange } from '@angular/material/select';
import { MatTable } from '@angular/material/table'; import { MatTable } from '@angular/material/table';
import { ActivatedRoute } from '@angular/router'; import { ActivatedRoute } from '@angular/router';
import { take, tap } from 'rxjs/operators'; import { take } from 'rxjs/operators';
import { ProjectMember, ProjectType, ProjectView, User } from 'src/app/proto/generated/management_pb'; import { ProjectGrantView, ProjectMember, ProjectType, ProjectView, User } from 'src/app/proto/generated/management_pb';
import { ProjectService } from 'src/app/services/project.service'; import { ProjectService } from 'src/app/services/project.service';
import { ToastService } from 'src/app/services/toast.service'; import { ToastService } from 'src/app/services/toast.service';
@ -18,10 +18,12 @@ import { ProjectMembersDataSource } from './project-members-datasource';
templateUrl: './project-members.component.html', templateUrl: './project-members.component.html',
styleUrls: ['./project-members.component.scss'], styleUrls: ['./project-members.component.scss'],
}) })
export class ProjectMembersComponent implements AfterViewInit { export class ProjectMembersComponent {
public project!: ProjectView.AsObject; public project!: ProjectView.AsObject | ProjectGrantView.AsObject;
public projectType: ProjectType = ProjectType.PROJECTTYPE_OWNED; public projectType: ProjectType = ProjectType.PROJECTTYPE_OWNED;
public disabled: boolean = false; public disabled: boolean = false;
public grantId: string = '';
public projectName: string = '';
@ViewChild(MatPaginator) public paginator!: MatPaginator; @ViewChild(MatPaginator) public paginator!: MatPaginator;
@ViewChild(MatTable) public table!: MatTable<ProjectMember.AsObject>; @ViewChild(MatTable) public table!: MatTable<ProjectMember.AsObject>;
public dataSource!: ProjectMembersDataSource; public dataSource!: ProjectMembersDataSource;
@ -42,23 +44,25 @@ export class ProjectMembersComponent implements AfterViewInit {
this.getRoleOptions(); this.getRoleOptions();
this.route.params.subscribe(params => { this.route.params.subscribe(params => {
this.projectService.GetProjectById(params.projectid).then(project => { this.grantId = params.grantid;
this.project = project.toObject(); if (this.projectType === ProjectType.PROJECTTYPE_OWNED) {
this.dataSource = new ProjectMembersDataSource(this.projectService); this.projectService.GetProjectById(params.projectid).then(project => {
this.dataSource.loadMembers(this.project, this.projectType, 0, 25, 'asc'); this.project = project.toObject();
}); this.projectName = this.project.name;
this.dataSource = new ProjectMembersDataSource(this.projectService);
this.dataSource.loadMembers(this.project.projectId, this.projectType, 0, 25);
});
} else if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) {
console.log(params.projectid, params.grantid);
this.projectService.GetGrantedProjectByID(params.projectid, params.grantid).then(project => {
this.project = project.toObject();
this.projectName = this.project.projectName;
this.dataSource = new ProjectMembersDataSource(this.projectService);
this.dataSource.loadMembers(this.project.projectId, this.projectType, 0, 25, this.grantId);
});
}
}); });
}); });
}
public ngAfterViewInit(): void {
this.paginator.page
.pipe(
tap(() => this.loadMembersPage()),
)
.subscribe();
} }
public getRoleOptions(): void { public getRoleOptions(): void {
@ -77,33 +81,25 @@ export class ProjectMembersComponent implements AfterViewInit {
} }
} }
private loadMembersPage(): void {
this.dataSource.loadMembers(
this.project,
this.projectType,
this.paginator.pageIndex,
this.paginator.pageSize,
);
}
public removeProjectMemberSelection(): void { public removeProjectMemberSelection(): void {
Promise.all(this.selection.selected.map(member => { Promise.all(this.selection.selected.map(member => {
return this.projectService.RemoveProjectMember(this.project.projectId, member.userId).then(() => { if (this.projectType === ProjectType.PROJECTTYPE_OWNED) {
this.toast.showInfo('PROJECT.TOAST.MEMBERREMOVED', true); return this.projectService.RemoveProjectMember(this.project.projectId, member.userId).then(() => {
}).catch(error => { this.toast.showInfo('PROJECT.TOAST.MEMBERREMOVED', true);
this.toast.showError(error); }).catch(error => {
}); this.toast.showError(error);
});
} else if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) {
return this.projectService.RemoveProjectGrantMember(this.project.projectId, this.grantId,
member.userId).then(() => {
this.toast.showInfo('PROJECT.TOAST.MEMBERREMOVED', true);
}).catch(error => {
this.toast.showError(error);
});
}
})); }));
} }
public removeMember(member: ProjectMember.AsObject): void {
this.projectService.RemoveProjectMember(this.project.projectId, member.userId).then(() => {
this.toast.showInfo('PROJECT.TOAST.MEMBERREMOVED', true);
}).catch(error => {
this.toast.showError(error);
});
}
public isAllSelected(): boolean { public isAllSelected(): boolean {
const numSelected = this.selection.selected.length; const numSelected = this.selection.selected.length;
const numRows = this.dataSource.membersSubject.value.length; const numRows = this.dataSource.membersSubject.value.length;
@ -132,7 +128,13 @@ export class ProjectMembersComponent implements AfterViewInit {
if (users && users.length && roles && roles.length) { if (users && users.length && roles && roles.length) {
Promise.all(users.map(user => { Promise.all(users.map(user => {
return this.projectService.AddProjectMember(this.project.projectId, user.id, roles); if (this.projectType === ProjectType.PROJECTTYPE_OWNED) {
return this.projectService.AddProjectMember(this.project.projectId, user.id, roles);
} else if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) {
return this.projectService.AddProjectGrantMember(this.project.projectId, this.grantId,
user.id, roles);
}
})).then(() => { })).then(() => {
this.toast.showInfo('PROJECT.TOAST.MEMBERSADDED', true); this.toast.showInfo('PROJECT.TOAST.MEMBERSADDED', true);
}).catch(error => { }).catch(error => {
@ -144,11 +146,21 @@ export class ProjectMembersComponent implements AfterViewInit {
} }
updateRoles(member: ProjectMember.AsObject, selectionChange: MatSelectChange): void { updateRoles(member: ProjectMember.AsObject, selectionChange: MatSelectChange): void {
this.projectService.ChangeProjectMember(this.project.projectId, member.userId, selectionChange.value) if (this.projectType === ProjectType.PROJECTTYPE_OWNED) {
.then((newmember: ProjectMember) => { this.projectService.ChangeProjectMember(this.project.projectId, member.userId, selectionChange.value)
this.toast.showInfo('PROJECT.TOAST.MEMBERADDED', true); .then((newmember: ProjectMember) => {
}).catch(error => { this.toast.showInfo('Member changed');
this.toast.showError(error); }).catch(error => {
}); this.toast.showError(error);
});
} else if (this.projectType === ProjectType.PROJECTTYPE_GRANTED) {
this.projectService.ChangeProjectGrantMember(this.project.projectId,
this.grantId, member.userId, selectionChange.value)
.then((newmember: ProjectMember) => {
this.toast.showInfo('Member changed');
}).catch(error => {
this.toast.showError(error);
});
}
} }
} }

View File

@ -33,9 +33,9 @@
<mat-tab-group mat-stretch-tabs class="tab-group" disablePagination="true"> <mat-tab-group mat-stretch-tabs class="tab-group" disablePagination="true">
<mat-tab label="Details"> <mat-tab label="Details">
<app-project-contributors *ngIf="project" <app-project-contributors *ngIf="project && grantId"
[disabled]="project?.state !== ProjectState.PROJECTSTATE_ACTIVE" [disabled]="project?.state !== ProjectState.PROJECTSTATE_ACTIVE"
[projectType]="ProjectType.PROJECTTYPE_GRANTED" [project]="project"> [projectType]="ProjectType.PROJECTTYPE_GRANTED" [project]="project" [grantId]="grantId">
</app-project-contributors> </app-project-contributors>
</mat-tab> </mat-tab>
<mat-tab label="{{ 'CHANGES.PROJECT.TITLE' | translate }}" class="flex-col"> <mat-tab label="{{ 'CHANGES.PROJECT.TITLE' | translate }}" class="flex-col">

View File

@ -22,7 +22,7 @@ const routes: Routes = [
}, },
}, },
{ {
path: ':projectid/members', path: ':projectid/grant/:grantid/members',
data: { data: {
type: ProjectType.PROJECTTYPE_GRANTED, type: ProjectType.PROJECTTYPE_GRANTED,
}, },

View File

@ -126,19 +126,19 @@ export class ProjectGrantMembersComponent implements AfterViewInit, OnInit {
dialogRef.afterClosed().subscribe((dataToAdd: ProjectGrantMembersCreateDialogExportType) => { dialogRef.afterClosed().subscribe((dataToAdd: ProjectGrantMembersCreateDialogExportType) => {
if (dataToAdd) { if (dataToAdd) {
dataToAdd.userIds.forEach((userid: string) => { Promise.all(dataToAdd.userIds.map((userid: string) => {
this.projectService.AddProjectGrantMember( return this.projectService.AddProjectGrantMember(
this.projectId, this.projectId,
this.grantId, this.grantId,
userid, userid,
dataToAdd.rolesKeyList, dataToAdd.rolesKeyList,
).then(() => { );
this.toast.showInfo('Project Grant Member successfully added!'); })).then(() => {
}).catch(error => { console.log('this');
this.toast.showError(error); this.toast.showInfo('Project Grant Member successfully added!');
}); }).catch(error => {
this.toast.showError(error);
}); });
} }
}); });
} }

View File

@ -3,5 +3,5 @@
"mgmtServiceUrl": "https://api.zitadel.dev", "mgmtServiceUrl": "https://api.zitadel.dev",
"adminServiceUrl":"https://api.zitadel.dev", "adminServiceUrl":"https://api.zitadel.dev",
"issuer": "https://issuer.zitadel.dev", "issuer": "https://issuer.zitadel.dev",
"clientid": "63146698922323188@zitadel" "clientid": "63426288794266821@zitadel"
} }