mirror of
https://github.com/bitwarden/browser.git
synced 2024-11-14 10:26:19 +01:00
[AC-2924] Remove GroupsComponentRefactor flag (#10259)
* Remove feature flag and old component
This commit is contained in:
parent
ad26f0890a
commit
c5c8c45bab
@ -1,7 +1,7 @@
|
|||||||
<app-header>
|
<app-header>
|
||||||
<bit-search
|
<bit-search
|
||||||
[placeholder]="'searchGroups' | i18n"
|
[placeholder]="'searchGroups' | i18n"
|
||||||
[(ngModel)]="searchText"
|
[formControl]="searchControl"
|
||||||
class="tw-w-80"
|
class="tw-w-80"
|
||||||
></bit-search>
|
></bit-search>
|
||||||
<button bitButton type="button" buttonType="primary" (click)="add()">
|
<button bitButton type="button" buttonType="primary" (click)="add()">
|
||||||
@ -18,15 +18,12 @@
|
|||||||
></i>
|
></i>
|
||||||
<span class="tw-sr-only">{{ "loading" | i18n }}</span>
|
<span class="tw-sr-only">{{ "loading" | i18n }}</span>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-container *ngIf="!loading && visibleGroups">
|
<ng-container *ngIf="!loading">
|
||||||
<p *ngIf="!visibleGroups.length">{{ "noGroupsInList" | i18n }}</p>
|
<p *ngIf="!dataSource.filteredData.length">{{ "noGroupsInList" | i18n }}</p>
|
||||||
<bit-table
|
<!-- The padding on the bottom of the cdk-virtual-scroll-viewport element is required to prevent table row content
|
||||||
*ngIf="visibleGroups.length"
|
from overflowing the <main> element. -->
|
||||||
infinite-scroll
|
<cdk-virtual-scroll-viewport scrollWindow [itemSize]="rowHeight" class="tw-pb-8">
|
||||||
[infiniteScrollDistance]="1"
|
<bit-table *ngIf="dataSource.filteredData.length" [dataSource]="dataSource">
|
||||||
[infiniteScrollDisabled]="!isPaging()"
|
|
||||||
(scrolled)="loadMore()"
|
|
||||||
>
|
|
||||||
<ng-container header>
|
<ng-container header>
|
||||||
<tr>
|
<tr>
|
||||||
<th bitCell class="tw-w-20">
|
<th bitCell class="tw-w-20">
|
||||||
@ -41,7 +38,7 @@
|
|||||||
"all" | i18n
|
"all" | i18n
|
||||||
}}</label>
|
}}</label>
|
||||||
</th>
|
</th>
|
||||||
<th bitCell>{{ "name" | i18n }}</th>
|
<th bitCell bitSortable="name" default>{{ "name" | i18n }}</th>
|
||||||
<th bitCell>{{ "collections" | i18n }}</th>
|
<th bitCell>{{ "collections" | i18n }}</th>
|
||||||
<th bitCell class="tw-w-10">
|
<th bitCell class="tw-w-10">
|
||||||
<button
|
<button
|
||||||
@ -62,8 +59,8 @@
|
|||||||
</th>
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-template body>
|
<ng-template body let-rows$>
|
||||||
<tr bitRow *ngFor="let g of visibleGroups">
|
<tr bitRow *cdkVirtualFor="let g of rows$" [ngClass]="rowHeightClass">
|
||||||
<td bitCell (click)="check(g)" class="tw-cursor-pointer">
|
<td bitCell (click)="check(g)" class="tw-cursor-pointer">
|
||||||
<input type="checkbox" bitCheckbox [(ngModel)]="g.checked" />
|
<input type="checkbox" bitCheckbox [(ngModel)]="g.checked" />
|
||||||
</td>
|
</td>
|
||||||
@ -108,5 +105,5 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</bit-table>
|
</bit-table>
|
||||||
|
</cdk-virtual-scroll-viewport>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-template #addEdit></ng-template>
|
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
import { Component, OnDestroy, OnInit, ViewChild, ViewContainerRef } from "@angular/core";
|
import { Component } from "@angular/core";
|
||||||
|
import { takeUntilDestroyed } from "@angular/core/rxjs-interop";
|
||||||
|
import { FormControl } from "@angular/forms";
|
||||||
import { ActivatedRoute } from "@angular/router";
|
import { ActivatedRoute } from "@angular/router";
|
||||||
import {
|
import {
|
||||||
BehaviorSubject,
|
BehaviorSubject,
|
||||||
@ -7,21 +9,15 @@ import {
|
|||||||
from,
|
from,
|
||||||
lastValueFrom,
|
lastValueFrom,
|
||||||
map,
|
map,
|
||||||
Subject,
|
|
||||||
switchMap,
|
switchMap,
|
||||||
takeUntil,
|
|
||||||
tap,
|
tap,
|
||||||
} from "rxjs";
|
} from "rxjs";
|
||||||
import { first } from "rxjs/operators";
|
import { debounceTime, first } from "rxjs/operators";
|
||||||
|
|
||||||
import { SearchPipe } from "@bitwarden/angular/pipes/search.pipe";
|
|
||||||
import { ApiService } from "@bitwarden/common/abstractions/api.service";
|
import { ApiService } from "@bitwarden/common/abstractions/api.service";
|
||||||
import { SearchService } from "@bitwarden/common/abstractions/search.service";
|
|
||||||
import { ListResponse } from "@bitwarden/common/models/response/list.response";
|
import { ListResponse } from "@bitwarden/common/models/response/list.response";
|
||||||
import { I18nService } from "@bitwarden/common/platform/abstractions/i18n.service";
|
import { I18nService } from "@bitwarden/common/platform/abstractions/i18n.service";
|
||||||
import { LogService } from "@bitwarden/common/platform/abstractions/log.service";
|
import { LogService } from "@bitwarden/common/platform/abstractions/log.service";
|
||||||
import { PlatformUtilsService } from "@bitwarden/common/platform/abstractions/platform-utils.service";
|
|
||||||
import { Utils } from "@bitwarden/common/platform/misc/utils";
|
|
||||||
import { CollectionService } from "@bitwarden/common/vault/abstractions/collection.service";
|
import { CollectionService } from "@bitwarden/common/vault/abstractions/collection.service";
|
||||||
import { CollectionData } from "@bitwarden/common/vault/models/data/collection.data";
|
import { CollectionData } from "@bitwarden/common/vault/models/data/collection.data";
|
||||||
import { Collection } from "@bitwarden/common/vault/models/domain/collection";
|
import { Collection } from "@bitwarden/common/vault/models/domain/collection";
|
||||||
@ -30,7 +26,7 @@ import {
|
|||||||
CollectionResponse,
|
CollectionResponse,
|
||||||
} from "@bitwarden/common/vault/models/response/collection.response";
|
} from "@bitwarden/common/vault/models/response/collection.response";
|
||||||
import { CollectionView } from "@bitwarden/common/vault/models/view/collection.view";
|
import { CollectionView } from "@bitwarden/common/vault/models/view/collection.view";
|
||||||
import { DialogService } from "@bitwarden/components";
|
import { DialogService, TableDataSource, ToastService } from "@bitwarden/components";
|
||||||
|
|
||||||
import { InternalGroupService as GroupService, GroupView } from "../core";
|
import { InternalGroupService as GroupService, GroupView } from "../core";
|
||||||
|
|
||||||
@ -40,21 +36,7 @@ import {
|
|||||||
openGroupAddEditDialog,
|
openGroupAddEditDialog,
|
||||||
} from "./group-add-edit.component";
|
} from "./group-add-edit.component";
|
||||||
|
|
||||||
type CollectionViewMap = {
|
|
||||||
[id: string]: CollectionView;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GroupDetailsRow = {
|
type GroupDetailsRow = {
|
||||||
/**
|
|
||||||
* Group Id (used for searching)
|
|
||||||
*/
|
|
||||||
id: string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Group name (used for searching)
|
|
||||||
*/
|
|
||||||
name: string;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Details used for displaying group information
|
* Details used for displaying group information
|
||||||
*/
|
*/
|
||||||
@ -72,59 +54,38 @@ type GroupDetailsRow = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated To be replaced with NewGroupsComponent which significantly refactors this component.
|
* Custom filter predicate that filters the groups table by id and name only.
|
||||||
* The GroupsComponentRefactor flag switches between the old and new components; this component will be removed when
|
* This is required because the default implementation searches by all properties, which can unintentionally match
|
||||||
* the feature flag is removed.
|
* with members' names (who are assigned to the group) or collection names (which the group has access to).
|
||||||
*/
|
*/
|
||||||
|
const groupsFilter = (filter: string) => {
|
||||||
|
const transformedFilter = filter.trim().toLowerCase();
|
||||||
|
return (data: GroupDetailsRow) => {
|
||||||
|
const group = data.details;
|
||||||
|
|
||||||
|
return (
|
||||||
|
group.id.toLowerCase().indexOf(transformedFilter) != -1 ||
|
||||||
|
group.name.toLowerCase().indexOf(transformedFilter) != -1
|
||||||
|
);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: "app-org-groups",
|
|
||||||
templateUrl: "groups.component.html",
|
templateUrl: "groups.component.html",
|
||||||
})
|
})
|
||||||
export class GroupsComponent implements OnInit, OnDestroy {
|
export class GroupsComponent {
|
||||||
@ViewChild("addEdit", { read: ViewContainerRef, static: true }) addEditModalRef: ViewContainerRef;
|
|
||||||
@ViewChild("usersTemplate", { read: ViewContainerRef, static: true })
|
|
||||||
usersModalRef: ViewContainerRef;
|
|
||||||
|
|
||||||
loading = true;
|
loading = true;
|
||||||
organizationId: string;
|
organizationId: string;
|
||||||
groups: GroupDetailsRow[];
|
|
||||||
|
|
||||||
protected didScroll = false;
|
protected dataSource = new TableDataSource<GroupDetailsRow>();
|
||||||
protected pageSize = 100;
|
protected searchControl = new FormControl("");
|
||||||
|
|
||||||
|
// Fixed sizes used for cdkVirtualScroll
|
||||||
|
protected rowHeight = 46;
|
||||||
|
protected rowHeightClass = `tw-h-[46px]`;
|
||||||
|
|
||||||
protected ModalTabType = GroupAddEditTabType;
|
protected ModalTabType = GroupAddEditTabType;
|
||||||
|
|
||||||
private pagedGroupsCount = 0;
|
|
||||||
private pagedGroups: GroupDetailsRow[];
|
|
||||||
private searchedGroups: GroupDetailsRow[];
|
|
||||||
private _searchText$ = new BehaviorSubject<string>("");
|
|
||||||
private destroy$ = new Subject<void>();
|
|
||||||
private refreshGroups$ = new BehaviorSubject<void>(null);
|
private refreshGroups$ = new BehaviorSubject<void>(null);
|
||||||
private isSearching: boolean = false;
|
|
||||||
|
|
||||||
get searchText() {
|
|
||||||
return this._searchText$.value;
|
|
||||||
}
|
|
||||||
set searchText(value: string) {
|
|
||||||
this._searchText$.next(value);
|
|
||||||
// Manually update as we are not using the search pipe in the template
|
|
||||||
this.updateSearchedGroups();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The list of groups that should be visible in the table.
|
|
||||||
* This is needed as there are two modes (paging/searching) and
|
|
||||||
* we need a reference to the currently visible groups for
|
|
||||||
* the Select All checkbox
|
|
||||||
*/
|
|
||||||
get visibleGroups(): GroupDetailsRow[] {
|
|
||||||
if (this.isPaging()) {
|
|
||||||
return this.pagedGroups;
|
|
||||||
}
|
|
||||||
if (this.isSearching) {
|
|
||||||
return this.searchedGroups;
|
|
||||||
}
|
|
||||||
return this.groups;
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private apiService: ApiService,
|
private apiService: ApiService,
|
||||||
@ -132,14 +93,10 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
private route: ActivatedRoute,
|
private route: ActivatedRoute,
|
||||||
private i18nService: I18nService,
|
private i18nService: I18nService,
|
||||||
private dialogService: DialogService,
|
private dialogService: DialogService,
|
||||||
private platformUtilsService: PlatformUtilsService,
|
|
||||||
private searchService: SearchService,
|
|
||||||
private logService: LogService,
|
private logService: LogService,
|
||||||
private collectionService: CollectionService,
|
private collectionService: CollectionService,
|
||||||
private searchPipe: SearchPipe,
|
private toastService: ToastService,
|
||||||
) {}
|
) {
|
||||||
|
|
||||||
async ngOnInit() {
|
|
||||||
this.route.params
|
this.route.params
|
||||||
.pipe(
|
.pipe(
|
||||||
tap((params) => (this.organizationId = params.organizationId)),
|
tap((params) => (this.organizationId = params.organizationId)),
|
||||||
@ -156,9 +113,7 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
]),
|
]),
|
||||||
),
|
),
|
||||||
map(([collectionMap, groups]) => {
|
map(([collectionMap, groups]) => {
|
||||||
return groups
|
return groups.map<GroupDetailsRow>((g) => ({
|
||||||
.sort(Utils.getSortFunction(this.i18nService, "name"))
|
|
||||||
.map<GroupDetailsRow>((g) => ({
|
|
||||||
id: g.id,
|
id: g.id,
|
||||||
name: g.name,
|
name: g.name,
|
||||||
details: g,
|
details: g,
|
||||||
@ -168,58 +123,23 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
.sort(this.i18nService.collator?.compare),
|
.sort(this.i18nService.collator?.compare),
|
||||||
}));
|
}));
|
||||||
}),
|
}),
|
||||||
takeUntil(this.destroy$),
|
takeUntilDestroyed(),
|
||||||
)
|
)
|
||||||
.subscribe((groups) => {
|
.subscribe((groups) => {
|
||||||
this.groups = groups;
|
this.dataSource.data = groups;
|
||||||
this.resetPaging();
|
|
||||||
this.updateSearchedGroups();
|
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.route.queryParams
|
// Connect the search input to the table dataSource filter input
|
||||||
.pipe(
|
this.searchControl.valueChanges
|
||||||
first(),
|
.pipe(debounceTime(200), takeUntilDestroyed())
|
||||||
concatMap(async (qParams) => {
|
.subscribe((v) => (this.dataSource.filter = groupsFilter(v)));
|
||||||
this.searchText = qParams.search;
|
|
||||||
}),
|
|
||||||
takeUntil(this.destroy$),
|
|
||||||
)
|
|
||||||
.subscribe();
|
|
||||||
|
|
||||||
this._searchText$
|
this.route.queryParams.pipe(first(), takeUntilDestroyed()).subscribe((qParams) => {
|
||||||
.pipe(
|
this.searchControl.setValue(qParams.search);
|
||||||
switchMap((searchText) => this.searchService.isSearchable(searchText)),
|
|
||||||
takeUntil(this.destroy$),
|
|
||||||
)
|
|
||||||
.subscribe((isSearchable) => {
|
|
||||||
this.isSearching = isSearchable;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
|
||||||
this.destroy$.next();
|
|
||||||
this.destroy$.complete();
|
|
||||||
}
|
|
||||||
|
|
||||||
loadMore() {
|
|
||||||
if (!this.groups || this.groups.length <= this.pageSize) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pagedLength = this.pagedGroups.length;
|
|
||||||
let pagedSize = this.pageSize;
|
|
||||||
if (pagedLength === 0 && this.pagedGroupsCount > this.pageSize) {
|
|
||||||
pagedSize = this.pagedGroupsCount;
|
|
||||||
}
|
|
||||||
if (this.groups.length > pagedLength) {
|
|
||||||
this.pagedGroups = this.pagedGroups.concat(
|
|
||||||
this.groups.slice(pagedLength, pagedLength + pagedSize),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
this.pagedGroupsCount = this.pagedGroups.length;
|
|
||||||
this.didScroll = this.pagedGroups.length > this.pageSize;
|
|
||||||
}
|
|
||||||
|
|
||||||
async edit(
|
async edit(
|
||||||
group: GroupDetailsRow,
|
group: GroupDetailsRow,
|
||||||
startingTabIndex: GroupAddEditTabType = GroupAddEditTabType.Info,
|
startingTabIndex: GroupAddEditTabType = GroupAddEditTabType.Info,
|
||||||
@ -237,14 +157,12 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
if (result == GroupAddEditDialogResultType.Saved) {
|
if (result == GroupAddEditDialogResultType.Saved) {
|
||||||
this.refreshGroups$.next();
|
this.refreshGroups$.next();
|
||||||
} else if (result == GroupAddEditDialogResultType.Deleted) {
|
} else if (result == GroupAddEditDialogResultType.Deleted) {
|
||||||
this.removeGroup(group.details.id);
|
this.removeGroup(group);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
add() {
|
async add() {
|
||||||
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
await this.edit(null);
|
||||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
||||||
this.edit(null);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async delete(groupRow: GroupDetailsRow) {
|
async delete(groupRow: GroupDetailsRow) {
|
||||||
@ -259,19 +177,19 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
await this.groupService.delete(this.organizationId, groupRow.details.id);
|
await this.groupService.delete(this.organizationId, groupRow.details.id);
|
||||||
this.platformUtilsService.showToast(
|
this.toastService.showToast({
|
||||||
"success",
|
variant: "success",
|
||||||
null,
|
title: null,
|
||||||
this.i18nService.t("deletedGroupId", groupRow.details.name),
|
message: this.i18nService.t("deletedGroupId", groupRow.details.name),
|
||||||
);
|
});
|
||||||
this.removeGroup(groupRow.details.id);
|
this.removeGroup(groupRow);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteAllSelected() {
|
async deleteAllSelected() {
|
||||||
const groupsToDelete = this.groups.filter((g) => g.checked);
|
const groupsToDelete = this.dataSource.data.filter((g) => g.checked);
|
||||||
|
|
||||||
if (groupsToDelete.length == 0) {
|
if (groupsToDelete.length == 0) {
|
||||||
return;
|
return;
|
||||||
@ -295,46 +213,31 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
this.organizationId,
|
this.organizationId,
|
||||||
groupsToDelete.map((g) => g.details.id),
|
groupsToDelete.map((g) => g.details.id),
|
||||||
);
|
);
|
||||||
this.platformUtilsService.showToast(
|
this.toastService.showToast({
|
||||||
"success",
|
variant: "success",
|
||||||
null,
|
title: null,
|
||||||
this.i18nService.t("deletedManyGroups", groupsToDelete.length.toString()),
|
message: this.i18nService.t("deletedManyGroups", groupsToDelete.length.toString()),
|
||||||
);
|
});
|
||||||
|
|
||||||
groupsToDelete.forEach((g) => this.removeGroup(g.details.id));
|
groupsToDelete.forEach((g) => this.removeGroup(g));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
resetPaging() {
|
|
||||||
this.pagedGroups = [];
|
|
||||||
this.loadMore();
|
|
||||||
}
|
|
||||||
|
|
||||||
check(groupRow: GroupDetailsRow) {
|
check(groupRow: GroupDetailsRow) {
|
||||||
groupRow.checked = !groupRow.checked;
|
groupRow.checked = !groupRow.checked;
|
||||||
}
|
}
|
||||||
|
|
||||||
toggleAllVisible(event: Event) {
|
toggleAllVisible(event: Event) {
|
||||||
this.visibleGroups.forEach((g) => (g.checked = (event.target as HTMLInputElement).checked));
|
this.dataSource.filteredData.forEach(
|
||||||
|
(g) => (g.checked = (event.target as HTMLInputElement).checked),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
isPaging() {
|
private removeGroup(groupRow: GroupDetailsRow) {
|
||||||
const searching = this.isSearching;
|
// Assign a new array to dataSource.data to trigger the setters and update the table
|
||||||
if (searching && this.didScroll) {
|
this.dataSource.data = this.dataSource.data.filter((g) => g !== groupRow);
|
||||||
this.resetPaging();
|
|
||||||
}
|
|
||||||
return !searching && this.groups && this.groups.length > this.pageSize;
|
|
||||||
}
|
|
||||||
|
|
||||||
private removeGroup(id: string) {
|
|
||||||
const index = this.groups.findIndex((g) => g.details.id === id);
|
|
||||||
if (index > -1) {
|
|
||||||
this.groups.splice(index, 1);
|
|
||||||
this.resetPaging();
|
|
||||||
this.updateSearchedGroups();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async toCollectionMap(response: ListResponse<CollectionResponse>) {
|
private async toCollectionMap(response: ListResponse<CollectionResponse>) {
|
||||||
@ -344,21 +247,9 @@ export class GroupsComponent implements OnInit, OnDestroy {
|
|||||||
const decryptedCollections = await this.collectionService.decryptMany(collections);
|
const decryptedCollections = await this.collectionService.decryptMany(collections);
|
||||||
|
|
||||||
// Convert to an object using collection Ids as keys for faster name lookups
|
// Convert to an object using collection Ids as keys for faster name lookups
|
||||||
const collectionMap: CollectionViewMap = {};
|
const collectionMap: Record<string, CollectionView> = {};
|
||||||
decryptedCollections.forEach((c) => (collectionMap[c.id] = c));
|
decryptedCollections.forEach((c) => (collectionMap[c.id] = c));
|
||||||
|
|
||||||
return collectionMap;
|
return collectionMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
private updateSearchedGroups() {
|
|
||||||
if (this.isSearching) {
|
|
||||||
// Making use of the pipe in the component as we need know which groups where filtered
|
|
||||||
this.searchedGroups = this.searchPipe.transform(
|
|
||||||
this.groups,
|
|
||||||
this.searchText,
|
|
||||||
(group) => group.details.name,
|
|
||||||
(group) => group.details.id,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,109 +0,0 @@
|
|||||||
<app-header>
|
|
||||||
<bit-search
|
|
||||||
[placeholder]="'searchGroups' | i18n"
|
|
||||||
[formControl]="searchControl"
|
|
||||||
class="tw-w-80"
|
|
||||||
></bit-search>
|
|
||||||
<button bitButton type="button" buttonType="primary" (click)="add()">
|
|
||||||
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
|
||||||
{{ "newGroup" | i18n }}
|
|
||||||
</button>
|
|
||||||
</app-header>
|
|
||||||
|
|
||||||
<ng-container *ngIf="loading">
|
|
||||||
<i
|
|
||||||
class="bwi bwi-spinner bwi-spin text-muted"
|
|
||||||
title="{{ 'loading' | i18n }}"
|
|
||||||
aria-hidden="true"
|
|
||||||
></i>
|
|
||||||
<span class="tw-sr-only">{{ "loading" | i18n }}</span>
|
|
||||||
</ng-container>
|
|
||||||
<ng-container *ngIf="!loading">
|
|
||||||
<p *ngIf="!dataSource.filteredData.length">{{ "noGroupsInList" | i18n }}</p>
|
|
||||||
<!-- The padding on the bottom of the cdk-virtual-scroll-viewport element is required to prevent table row content
|
|
||||||
from overflowing the <main> element. -->
|
|
||||||
<cdk-virtual-scroll-viewport scrollWindow [itemSize]="rowHeight" class="tw-pb-8">
|
|
||||||
<bit-table *ngIf="dataSource.filteredData.length" [dataSource]="dataSource">
|
|
||||||
<ng-container header>
|
|
||||||
<tr>
|
|
||||||
<th bitCell class="tw-w-20">
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
bitCheckbox
|
|
||||||
class="tw-mr-2"
|
|
||||||
(change)="toggleAllVisible($event)"
|
|
||||||
id="selectAll"
|
|
||||||
/>
|
|
||||||
<label class="tw-mb-0 !tw-font-bold !tw-text-muted" for="selectAll">{{
|
|
||||||
"all" | i18n
|
|
||||||
}}</label>
|
|
||||||
</th>
|
|
||||||
<th bitCell bitSortable="name" default>{{ "name" | i18n }}</th>
|
|
||||||
<th bitCell>{{ "collections" | i18n }}</th>
|
|
||||||
<th bitCell class="tw-w-10">
|
|
||||||
<button
|
|
||||||
[bitMenuTriggerFor]="headerMenu"
|
|
||||||
type="button"
|
|
||||||
bitIconButton="bwi-ellipsis-v"
|
|
||||||
size="small"
|
|
||||||
appA11yTitle="{{ 'options' | i18n }}"
|
|
||||||
></button>
|
|
||||||
|
|
||||||
<bit-menu #headerMenu>
|
|
||||||
<button type="button" bitMenuItem (click)="deleteAllSelected()">
|
|
||||||
<span class="tw-text-danger"
|
|
||||||
><i aria-hidden="true" class="bwi bwi-trash"></i> {{ "delete" | i18n }}</span
|
|
||||||
>
|
|
||||||
</button>
|
|
||||||
</bit-menu>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</ng-container>
|
|
||||||
<ng-template body let-rows$>
|
|
||||||
<tr bitRow *cdkVirtualFor="let g of rows$" [ngClass]="rowHeightClass">
|
|
||||||
<td bitCell (click)="check(g)" class="tw-cursor-pointer">
|
|
||||||
<input type="checkbox" bitCheckbox [(ngModel)]="g.checked" />
|
|
||||||
</td>
|
|
||||||
<td bitCell class="tw-cursor-pointer tw-font-bold" (click)="edit(g)">
|
|
||||||
<button type="button" bitLink>
|
|
||||||
{{ g.details.name }}
|
|
||||||
</button>
|
|
||||||
</td>
|
|
||||||
<td bitCell (click)="edit(g, ModalTabType.Collections)" class="tw-cursor-pointer">
|
|
||||||
<bit-badge-list
|
|
||||||
[items]="g.collectionNames"
|
|
||||||
[maxItems]="3"
|
|
||||||
variant="secondary"
|
|
||||||
></bit-badge-list>
|
|
||||||
</td>
|
|
||||||
<td bitCell>
|
|
||||||
<button
|
|
||||||
[bitMenuTriggerFor]="rowMenu"
|
|
||||||
type="button"
|
|
||||||
bitIconButton="bwi-ellipsis-v"
|
|
||||||
size="small"
|
|
||||||
appA11yTitle="{{ 'options' | i18n }}"
|
|
||||||
></button>
|
|
||||||
|
|
||||||
<bit-menu #rowMenu>
|
|
||||||
<button type="button" bitMenuItem (click)="edit(g)">
|
|
||||||
<i aria-hidden="true" class="bwi bwi-pencil-square"></i> {{ "editInfo" | i18n }}
|
|
||||||
</button>
|
|
||||||
<button type="button" bitMenuItem (click)="edit(g, ModalTabType.Members)">
|
|
||||||
<i aria-hidden="true" class="bwi bwi-user"></i> {{ "members" | i18n }}
|
|
||||||
</button>
|
|
||||||
<button type="button" bitMenuItem (click)="edit(g, ModalTabType.Collections)">
|
|
||||||
<i aria-hidden="true" class="bwi bwi-collection"></i> {{ "collections" | i18n }}
|
|
||||||
</button>
|
|
||||||
<button type="button" bitMenuItem (click)="delete(g)">
|
|
||||||
<span class="tw-text-danger"
|
|
||||||
><i aria-hidden="true" class="bwi bwi-trash"></i> {{ "delete" | i18n }}</span
|
|
||||||
>
|
|
||||||
</button>
|
|
||||||
</bit-menu>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</ng-template>
|
|
||||||
</bit-table>
|
|
||||||
</cdk-virtual-scroll-viewport>
|
|
||||||
</ng-container>
|
|
@ -1,255 +0,0 @@
|
|||||||
import { Component } from "@angular/core";
|
|
||||||
import { takeUntilDestroyed } from "@angular/core/rxjs-interop";
|
|
||||||
import { FormControl } from "@angular/forms";
|
|
||||||
import { ActivatedRoute } from "@angular/router";
|
|
||||||
import {
|
|
||||||
BehaviorSubject,
|
|
||||||
combineLatest,
|
|
||||||
concatMap,
|
|
||||||
from,
|
|
||||||
lastValueFrom,
|
|
||||||
map,
|
|
||||||
switchMap,
|
|
||||||
tap,
|
|
||||||
} from "rxjs";
|
|
||||||
import { debounceTime, first } from "rxjs/operators";
|
|
||||||
|
|
||||||
import { ApiService } from "@bitwarden/common/abstractions/api.service";
|
|
||||||
import { ListResponse } from "@bitwarden/common/models/response/list.response";
|
|
||||||
import { I18nService } from "@bitwarden/common/platform/abstractions/i18n.service";
|
|
||||||
import { LogService } from "@bitwarden/common/platform/abstractions/log.service";
|
|
||||||
import { CollectionService } from "@bitwarden/common/vault/abstractions/collection.service";
|
|
||||||
import { CollectionData } from "@bitwarden/common/vault/models/data/collection.data";
|
|
||||||
import { Collection } from "@bitwarden/common/vault/models/domain/collection";
|
|
||||||
import {
|
|
||||||
CollectionDetailsResponse,
|
|
||||||
CollectionResponse,
|
|
||||||
} from "@bitwarden/common/vault/models/response/collection.response";
|
|
||||||
import { CollectionView } from "@bitwarden/common/vault/models/view/collection.view";
|
|
||||||
import { DialogService, TableDataSource, ToastService } from "@bitwarden/components";
|
|
||||||
|
|
||||||
import { InternalGroupService as GroupService, GroupView } from "../core";
|
|
||||||
|
|
||||||
import {
|
|
||||||
GroupAddEditDialogResultType,
|
|
||||||
GroupAddEditTabType,
|
|
||||||
openGroupAddEditDialog,
|
|
||||||
} from "./group-add-edit.component";
|
|
||||||
|
|
||||||
type GroupDetailsRow = {
|
|
||||||
/**
|
|
||||||
* Details used for displaying group information
|
|
||||||
*/
|
|
||||||
details: GroupView;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* True if the group is selected in the table
|
|
||||||
*/
|
|
||||||
checked?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A list of collection names the group has access to
|
|
||||||
*/
|
|
||||||
collectionNames?: string[];
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Custom filter predicate that filters the groups table by id and name only.
|
|
||||||
* This is required because the default implementation searches by all properties, which can unintentionally match
|
|
||||||
* with members' names (who are assigned to the group) or collection names (which the group has access to).
|
|
||||||
*/
|
|
||||||
const groupsFilter = (filter: string) => {
|
|
||||||
const transformedFilter = filter.trim().toLowerCase();
|
|
||||||
return (data: GroupDetailsRow) => {
|
|
||||||
const group = data.details;
|
|
||||||
|
|
||||||
return (
|
|
||||||
group.id.toLowerCase().indexOf(transformedFilter) != -1 ||
|
|
||||||
group.name.toLowerCase().indexOf(transformedFilter) != -1
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
templateUrl: "new-groups.component.html",
|
|
||||||
})
|
|
||||||
export class NewGroupsComponent {
|
|
||||||
loading = true;
|
|
||||||
organizationId: string;
|
|
||||||
|
|
||||||
protected dataSource = new TableDataSource<GroupDetailsRow>();
|
|
||||||
protected searchControl = new FormControl("");
|
|
||||||
|
|
||||||
// Fixed sizes used for cdkVirtualScroll
|
|
||||||
protected rowHeight = 46;
|
|
||||||
protected rowHeightClass = `tw-h-[46px]`;
|
|
||||||
|
|
||||||
protected ModalTabType = GroupAddEditTabType;
|
|
||||||
private refreshGroups$ = new BehaviorSubject<void>(null);
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
private apiService: ApiService,
|
|
||||||
private groupService: GroupService,
|
|
||||||
private route: ActivatedRoute,
|
|
||||||
private i18nService: I18nService,
|
|
||||||
private dialogService: DialogService,
|
|
||||||
private logService: LogService,
|
|
||||||
private collectionService: CollectionService,
|
|
||||||
private toastService: ToastService,
|
|
||||||
) {
|
|
||||||
this.route.params
|
|
||||||
.pipe(
|
|
||||||
tap((params) => (this.organizationId = params.organizationId)),
|
|
||||||
switchMap(() =>
|
|
||||||
combineLatest([
|
|
||||||
// collectionMap
|
|
||||||
from(this.apiService.getCollections(this.organizationId)).pipe(
|
|
||||||
concatMap((response) => this.toCollectionMap(response)),
|
|
||||||
),
|
|
||||||
// groups
|
|
||||||
this.refreshGroups$.pipe(
|
|
||||||
switchMap(() => this.groupService.getAll(this.organizationId)),
|
|
||||||
),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
map(([collectionMap, groups]) => {
|
|
||||||
return groups.map<GroupDetailsRow>((g) => ({
|
|
||||||
id: g.id,
|
|
||||||
name: g.name,
|
|
||||||
details: g,
|
|
||||||
checked: false,
|
|
||||||
collectionNames: g.collections
|
|
||||||
.map((c) => collectionMap[c.id]?.name)
|
|
||||||
.sort(this.i18nService.collator?.compare),
|
|
||||||
}));
|
|
||||||
}),
|
|
||||||
takeUntilDestroyed(),
|
|
||||||
)
|
|
||||||
.subscribe((groups) => {
|
|
||||||
this.dataSource.data = groups;
|
|
||||||
this.loading = false;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Connect the search input to the table dataSource filter input
|
|
||||||
this.searchControl.valueChanges
|
|
||||||
.pipe(debounceTime(200), takeUntilDestroyed())
|
|
||||||
.subscribe((v) => (this.dataSource.filter = groupsFilter(v)));
|
|
||||||
|
|
||||||
this.route.queryParams.pipe(first(), takeUntilDestroyed()).subscribe((qParams) => {
|
|
||||||
this.searchControl.setValue(qParams.search);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async edit(
|
|
||||||
group: GroupDetailsRow,
|
|
||||||
startingTabIndex: GroupAddEditTabType = GroupAddEditTabType.Info,
|
|
||||||
) {
|
|
||||||
const dialogRef = openGroupAddEditDialog(this.dialogService, {
|
|
||||||
data: {
|
|
||||||
initialTab: startingTabIndex,
|
|
||||||
organizationId: this.organizationId,
|
|
||||||
groupId: group != null ? group.details.id : null,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const result = await lastValueFrom(dialogRef.closed);
|
|
||||||
|
|
||||||
if (result == GroupAddEditDialogResultType.Saved) {
|
|
||||||
this.refreshGroups$.next();
|
|
||||||
} else if (result == GroupAddEditDialogResultType.Deleted) {
|
|
||||||
this.removeGroup(group);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async add() {
|
|
||||||
await this.edit(null);
|
|
||||||
}
|
|
||||||
|
|
||||||
async delete(groupRow: GroupDetailsRow) {
|
|
||||||
const confirmed = await this.dialogService.openSimpleDialog({
|
|
||||||
title: groupRow.details.name,
|
|
||||||
content: { key: "deleteGroupConfirmation" },
|
|
||||||
type: "warning",
|
|
||||||
});
|
|
||||||
if (!confirmed) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await this.groupService.delete(this.organizationId, groupRow.details.id);
|
|
||||||
this.toastService.showToast({
|
|
||||||
variant: "success",
|
|
||||||
title: null,
|
|
||||||
message: this.i18nService.t("deletedGroupId", groupRow.details.name),
|
|
||||||
});
|
|
||||||
this.removeGroup(groupRow);
|
|
||||||
} catch (e) {
|
|
||||||
this.logService.error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async deleteAllSelected() {
|
|
||||||
const groupsToDelete = this.dataSource.data.filter((g) => g.checked);
|
|
||||||
|
|
||||||
if (groupsToDelete.length == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const deleteMessage = groupsToDelete.map((g) => g.details.name).join(", ");
|
|
||||||
const confirmed = await this.dialogService.openSimpleDialog({
|
|
||||||
title: {
|
|
||||||
key: "deleteMultipleGroupsConfirmation",
|
|
||||||
placeholders: [groupsToDelete.length.toString()],
|
|
||||||
},
|
|
||||||
content: deleteMessage,
|
|
||||||
type: "warning",
|
|
||||||
});
|
|
||||||
if (!confirmed) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await this.groupService.deleteMany(
|
|
||||||
this.organizationId,
|
|
||||||
groupsToDelete.map((g) => g.details.id),
|
|
||||||
);
|
|
||||||
this.toastService.showToast({
|
|
||||||
variant: "success",
|
|
||||||
title: null,
|
|
||||||
message: this.i18nService.t("deletedManyGroups", groupsToDelete.length.toString()),
|
|
||||||
});
|
|
||||||
|
|
||||||
groupsToDelete.forEach((g) => this.removeGroup(g));
|
|
||||||
} catch (e) {
|
|
||||||
this.logService.error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
check(groupRow: GroupDetailsRow) {
|
|
||||||
groupRow.checked = !groupRow.checked;
|
|
||||||
}
|
|
||||||
|
|
||||||
toggleAllVisible(event: Event) {
|
|
||||||
this.dataSource.filteredData.forEach(
|
|
||||||
(g) => (g.checked = (event.target as HTMLInputElement).checked),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private removeGroup(groupRow: GroupDetailsRow) {
|
|
||||||
// Assign a new array to dataSource.data to trigger the setters and update the table
|
|
||||||
this.dataSource.data = this.dataSource.data.filter((g) => g !== groupRow);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async toCollectionMap(response: ListResponse<CollectionResponse>) {
|
|
||||||
const collections = response.data.map(
|
|
||||||
(r) => new Collection(new CollectionData(r as CollectionDetailsResponse)),
|
|
||||||
);
|
|
||||||
const decryptedCollections = await this.collectionService.decryptMany(collections);
|
|
||||||
|
|
||||||
// Convert to an object using collection Ids as keys for faster name lookups
|
|
||||||
const collectionMap: Record<string, CollectionView> = {};
|
|
||||||
decryptedCollections.forEach((c) => (collectionMap[c.id] = c));
|
|
||||||
|
|
||||||
return collectionMap;
|
|
||||||
}
|
|
||||||
}
|
|
@ -2,7 +2,6 @@ import { NgModule } from "@angular/core";
|
|||||||
import { RouterModule, Routes } from "@angular/router";
|
import { RouterModule, Routes } from "@angular/router";
|
||||||
|
|
||||||
import { authGuard } from "@bitwarden/angular/auth/guards";
|
import { authGuard } from "@bitwarden/angular/auth/guards";
|
||||||
import { featureFlaggedRoute } from "@bitwarden/angular/platform/utils/feature-flagged-route";
|
|
||||||
import {
|
import {
|
||||||
canAccessOrgAdmin,
|
canAccessOrgAdmin,
|
||||||
canAccessGroupsTab,
|
canAccessGroupsTab,
|
||||||
@ -12,16 +11,15 @@ import {
|
|||||||
canAccessSettingsTab,
|
canAccessSettingsTab,
|
||||||
} from "@bitwarden/common/admin-console/abstractions/organization/organization.service.abstraction";
|
} from "@bitwarden/common/admin-console/abstractions/organization/organization.service.abstraction";
|
||||||
import { Organization } from "@bitwarden/common/admin-console/models/domain/organization";
|
import { Organization } from "@bitwarden/common/admin-console/models/domain/organization";
|
||||||
import { FeatureFlag } from "@bitwarden/common/enums/feature-flag.enum";
|
|
||||||
|
|
||||||
import { organizationPermissionsGuard } from "../../admin-console/organizations/guards/org-permissions.guard";
|
import { organizationPermissionsGuard } from "../../admin-console/organizations/guards/org-permissions.guard";
|
||||||
import { organizationRedirectGuard } from "../../admin-console/organizations/guards/org-redirect.guard";
|
import { organizationRedirectGuard } from "../../admin-console/organizations/guards/org-redirect.guard";
|
||||||
import { OrganizationLayoutComponent } from "../../admin-console/organizations/layouts/organization-layout.component";
|
import { OrganizationLayoutComponent } from "../../admin-console/organizations/layouts/organization-layout.component";
|
||||||
import { GroupsComponent } from "../../admin-console/organizations/manage/groups.component";
|
|
||||||
import { NewGroupsComponent } from "../../admin-console/organizations/manage/new-groups.component";
|
|
||||||
import { deepLinkGuard } from "../../auth/guards/deep-link.guard";
|
import { deepLinkGuard } from "../../auth/guards/deep-link.guard";
|
||||||
import { VaultModule } from "../../vault/org-vault/vault.module";
|
import { VaultModule } from "../../vault/org-vault/vault.module";
|
||||||
|
|
||||||
|
import { GroupsComponent } from "./manage/groups.component";
|
||||||
|
|
||||||
const routes: Routes = [
|
const routes: Routes = [
|
||||||
{
|
{
|
||||||
path: ":organizationId",
|
path: ":organizationId",
|
||||||
@ -49,18 +47,14 @@ const routes: Routes = [
|
|||||||
path: "members",
|
path: "members",
|
||||||
loadChildren: () => import("./members").then((m) => m.MembersModule),
|
loadChildren: () => import("./members").then((m) => m.MembersModule),
|
||||||
},
|
},
|
||||||
...featureFlaggedRoute({
|
{
|
||||||
defaultComponent: GroupsComponent,
|
component: GroupsComponent,
|
||||||
flaggedComponent: NewGroupsComponent,
|
|
||||||
featureFlag: FeatureFlag.GroupsComponentRefactor,
|
|
||||||
routeOptions: {
|
|
||||||
path: "groups",
|
path: "groups",
|
||||||
canActivate: [organizationPermissionsGuard(canAccessGroupsTab)],
|
canActivate: [organizationPermissionsGuard(canAccessGroupsTab)],
|
||||||
data: {
|
data: {
|
||||||
titleId: "groups",
|
titleId: "groups",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
|
||||||
{
|
{
|
||||||
path: "reporting",
|
path: "reporting",
|
||||||
loadChildren: () =>
|
loadChildren: () =>
|
||||||
|
@ -6,7 +6,6 @@ import { LooseComponentsModule } from "../../shared";
|
|||||||
import { CoreOrganizationModule } from "./core";
|
import { CoreOrganizationModule } from "./core";
|
||||||
import { GroupAddEditComponent } from "./manage/group-add-edit.component";
|
import { GroupAddEditComponent } from "./manage/group-add-edit.component";
|
||||||
import { GroupsComponent } from "./manage/groups.component";
|
import { GroupsComponent } from "./manage/groups.component";
|
||||||
import { NewGroupsComponent } from "./manage/new-groups.component";
|
|
||||||
import { OrganizationsRoutingModule } from "./organization-routing.module";
|
import { OrganizationsRoutingModule } from "./organization-routing.module";
|
||||||
import { SharedOrganizationModule } from "./shared";
|
import { SharedOrganizationModule } from "./shared";
|
||||||
import { AccessSelectorModule } from "./shared/components/access-selector";
|
import { AccessSelectorModule } from "./shared/components/access-selector";
|
||||||
@ -20,6 +19,6 @@ import { AccessSelectorModule } from "./shared/components/access-selector";
|
|||||||
LooseComponentsModule,
|
LooseComponentsModule,
|
||||||
ScrollingModule,
|
ScrollingModule,
|
||||||
],
|
],
|
||||||
declarations: [GroupsComponent, NewGroupsComponent, GroupAddEditComponent],
|
declarations: [GroupsComponent, GroupAddEditComponent],
|
||||||
})
|
})
|
||||||
export class OrganizationModule {}
|
export class OrganizationModule {}
|
||||||
|
@ -22,7 +22,6 @@ export enum FeatureFlag {
|
|||||||
TwoFactorComponentRefactor = "two-factor-component-refactor",
|
TwoFactorComponentRefactor = "two-factor-component-refactor",
|
||||||
EnableTimeThreshold = "PM-5864-dollar-threshold",
|
EnableTimeThreshold = "PM-5864-dollar-threshold",
|
||||||
InlineMenuPositioningImprovements = "inline-menu-positioning-improvements",
|
InlineMenuPositioningImprovements = "inline-menu-positioning-improvements",
|
||||||
GroupsComponentRefactor = "groups-component-refactor",
|
|
||||||
ProviderClientVaultPrivacyBanner = "ac-2833-provider-client-vault-privacy-banner",
|
ProviderClientVaultPrivacyBanner = "ac-2833-provider-client-vault-privacy-banner",
|
||||||
VaultBulkManagementAction = "vault-bulk-management-action",
|
VaultBulkManagementAction = "vault-bulk-management-action",
|
||||||
AC2828_ProviderPortalMembersPage = "AC-2828_provider-portal-members-page",
|
AC2828_ProviderPortalMembersPage = "AC-2828_provider-portal-members-page",
|
||||||
@ -60,7 +59,6 @@ export const DefaultFeatureFlagValue = {
|
|||||||
[FeatureFlag.TwoFactorComponentRefactor]: FALSE,
|
[FeatureFlag.TwoFactorComponentRefactor]: FALSE,
|
||||||
[FeatureFlag.EnableTimeThreshold]: FALSE,
|
[FeatureFlag.EnableTimeThreshold]: FALSE,
|
||||||
[FeatureFlag.InlineMenuPositioningImprovements]: FALSE,
|
[FeatureFlag.InlineMenuPositioningImprovements]: FALSE,
|
||||||
[FeatureFlag.GroupsComponentRefactor]: FALSE,
|
|
||||||
[FeatureFlag.ProviderClientVaultPrivacyBanner]: FALSE,
|
[FeatureFlag.ProviderClientVaultPrivacyBanner]: FALSE,
|
||||||
[FeatureFlag.VaultBulkManagementAction]: FALSE,
|
[FeatureFlag.VaultBulkManagementAction]: FALSE,
|
||||||
[FeatureFlag.AC2828_ProviderPortalMembersPage]: FALSE,
|
[FeatureFlag.AC2828_ProviderPortalMembersPage]: FALSE,
|
||||||
|
Loading…
Reference in New Issue
Block a user