mirror of
https://github.com/bitwarden/browser
synced 2025-12-15 07:43:35 +00:00
This PR introduces a generic `DialogService` which can be used by all the clients. This allows us to decouple dialogs from the `PlatformUtilsHelper`. The `DialogService` provides a new method, `openSimpleDialog` which is the new interface for that type of dialogs. This gives us 3 different implementations: - Web: DialogService modern dialogs - Browser: SweetAlert - Desktop: Native electron based
152 lines
4.9 KiB
TypeScript
152 lines
4.9 KiB
TypeScript
import { DialogConfig, DialogRef, DIALOG_DATA } from "@angular/cdk/dialog";
|
|
import { Component, Inject, OnInit } from "@angular/core";
|
|
|
|
import { DialogServiceAbstraction } from "@bitwarden/angular/services/dialog";
|
|
import { I18nService } from "@bitwarden/common/abstractions/i18n.service";
|
|
import { LogService } from "@bitwarden/common/abstractions/log.service";
|
|
import { PlatformUtilsService } from "@bitwarden/common/abstractions/platformUtils.service";
|
|
import { CollectionService } from "@bitwarden/common/admin-console/abstractions/collection.service";
|
|
import { OrganizationService } from "@bitwarden/common/admin-console/abstractions/organization/organization.service.abstraction";
|
|
import { Organization } from "@bitwarden/common/admin-console/models/domain/organization";
|
|
import { CollectionView } from "@bitwarden/common/admin-console/models/view/collection.view";
|
|
import { Checkable, isChecked } from "@bitwarden/common/types/checkable";
|
|
import { CipherService } from "@bitwarden/common/vault/abstractions/cipher.service";
|
|
import { CipherView } from "@bitwarden/common/vault/models/view/cipher.view";
|
|
|
|
export interface BulkShareDialogParams {
|
|
ciphers: CipherView[];
|
|
organizationId?: string;
|
|
}
|
|
|
|
export enum BulkShareDialogResult {
|
|
Shared = "shared",
|
|
Canceled = "canceled",
|
|
}
|
|
|
|
/**
|
|
* Strongly typed helper to open a BulkShareDialog
|
|
* @param dialogService Instance of the dialog service that will be used to open the dialog
|
|
* @param config Configuration for the dialog
|
|
*/
|
|
export const openBulkShareDialog = (
|
|
dialogService: DialogServiceAbstraction,
|
|
config: DialogConfig<BulkShareDialogParams>
|
|
) => {
|
|
return dialogService.open<BulkShareDialogResult, BulkShareDialogParams>(
|
|
BulkShareDialogComponent,
|
|
config
|
|
);
|
|
};
|
|
|
|
@Component({
|
|
templateUrl: "bulk-share-dialog.component.html",
|
|
})
|
|
export class BulkShareDialogComponent implements OnInit {
|
|
ciphers: CipherView[] = [];
|
|
organizationId: string;
|
|
|
|
nonShareableCount = 0;
|
|
collections: Checkable<CollectionView>[] = [];
|
|
organizations: Organization[] = [];
|
|
shareableCiphers: CipherView[] = [];
|
|
|
|
private writeableCollections: CollectionView[] = [];
|
|
|
|
constructor(
|
|
@Inject(DIALOG_DATA) params: BulkShareDialogParams,
|
|
private dialogRef: DialogRef<BulkShareDialogResult>,
|
|
private cipherService: CipherService,
|
|
private platformUtilsService: PlatformUtilsService,
|
|
private i18nService: I18nService,
|
|
private collectionService: CollectionService,
|
|
private organizationService: OrganizationService,
|
|
private logService: LogService
|
|
) {
|
|
this.ciphers = params.ciphers ?? [];
|
|
this.organizationId = params.organizationId;
|
|
}
|
|
|
|
async ngOnInit() {
|
|
this.shareableCiphers = this.ciphers.filter(
|
|
(c) => !c.hasOldAttachments && c.organizationId == null
|
|
);
|
|
this.nonShareableCount = this.ciphers.length - this.shareableCiphers.length;
|
|
const allCollections = await this.collectionService.getAllDecrypted();
|
|
this.writeableCollections = allCollections.filter((c) => !c.readOnly);
|
|
this.organizations = await this.organizationService.getAll();
|
|
if (this.organizationId == null && this.organizations.length > 0) {
|
|
this.organizationId = this.organizations[0].id;
|
|
}
|
|
this.filterCollections();
|
|
}
|
|
|
|
ngOnDestroy() {
|
|
this.selectAll(false);
|
|
}
|
|
|
|
filterCollections() {
|
|
this.selectAll(false);
|
|
if (this.organizationId == null || this.writeableCollections.length === 0) {
|
|
this.collections = [];
|
|
} else {
|
|
this.collections = this.writeableCollections.filter(
|
|
(c) => c.organizationId === this.organizationId
|
|
);
|
|
}
|
|
}
|
|
|
|
submit = async () => {
|
|
const checkedCollectionIds = this.collections.filter(isChecked).map((c) => c.id);
|
|
try {
|
|
await this.cipherService.shareManyWithServer(
|
|
this.shareableCiphers,
|
|
this.organizationId,
|
|
checkedCollectionIds
|
|
);
|
|
const orgName =
|
|
this.organizations.find((o) => o.id === this.organizationId)?.name ??
|
|
this.i18nService.t("organization");
|
|
this.platformUtilsService.showToast(
|
|
"success",
|
|
null,
|
|
this.i18nService.t("movedItemsToOrg", orgName)
|
|
);
|
|
this.close(BulkShareDialogResult.Shared);
|
|
} catch (e) {
|
|
this.logService.error(e);
|
|
}
|
|
};
|
|
|
|
check(c: Checkable<CollectionView>, select?: boolean) {
|
|
c.checked = select == null ? !c.checked : select;
|
|
}
|
|
|
|
selectAll(select: boolean) {
|
|
const collections = select ? this.collections : this.writeableCollections;
|
|
collections.forEach((c) => this.check(c, select));
|
|
}
|
|
|
|
get canSave() {
|
|
if (
|
|
this.shareableCiphers != null &&
|
|
this.shareableCiphers.length > 0 &&
|
|
this.collections != null
|
|
) {
|
|
for (let i = 0; i < this.collections.length; i++) {
|
|
if (this.collections[i].checked) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected cancel() {
|
|
this.close(BulkShareDialogResult.Canceled);
|
|
}
|
|
|
|
private close(result: BulkShareDialogResult) {
|
|
this.dialogRef.close(result);
|
|
}
|
|
}
|