mirror of
https://github.com/bitwarden/browser
synced 2025-12-10 13:23:34 +00:00
188 lines
6.6 KiB
TypeScript
188 lines
6.6 KiB
TypeScript
// FIXME: Update this file to be type safe and remove this and next line
|
|
// @ts-strict-ignore
|
|
import { CommonModule } from "@angular/common";
|
|
import { Component, NgZone, OnDestroy, OnInit, ViewChild } from "@angular/core";
|
|
import { FormsModule } from "@angular/forms";
|
|
|
|
import { JslibModule } from "@bitwarden/angular/jslib.module";
|
|
import { SendComponent as BaseSendComponent } from "@bitwarden/angular/tools/send/send.component";
|
|
import { PolicyService } from "@bitwarden/common/admin-console/abstractions/policy/policy.service.abstraction";
|
|
import { AccountService } from "@bitwarden/common/auth/abstractions/account.service";
|
|
import { BroadcasterService } from "@bitwarden/common/platform/abstractions/broadcaster.service";
|
|
import { EnvironmentService } from "@bitwarden/common/platform/abstractions/environment.service";
|
|
import { I18nService } from "@bitwarden/common/platform/abstractions/i18n.service";
|
|
import { LogService } from "@bitwarden/common/platform/abstractions/log.service";
|
|
import { PlatformUtilsService } from "@bitwarden/common/platform/abstractions/platform-utils.service";
|
|
import { SendView } from "@bitwarden/common/tools/send/models/view/send.view";
|
|
import { SendApiService } from "@bitwarden/common/tools/send/services/send-api.service.abstraction";
|
|
import { SendService } from "@bitwarden/common/tools/send/services/send.service.abstraction";
|
|
import { SearchService } from "@bitwarden/common/vault/abstractions/search.service";
|
|
import { DialogService, ToastService } from "@bitwarden/components";
|
|
|
|
import { invokeMenu, RendererMenuItem } from "../../../utils";
|
|
import { NavComponent } from "../../layout/nav.component";
|
|
import { SearchBarService } from "../../layout/search/search-bar.service";
|
|
|
|
import { AddEditComponent } from "./add-edit.component";
|
|
|
|
// FIXME: update to use a const object instead of a typescript enum
|
|
// eslint-disable-next-line @bitwarden/platform/no-enums
|
|
enum Action {
|
|
None = "",
|
|
Add = "add",
|
|
Edit = "edit",
|
|
}
|
|
|
|
const BroadcasterSubscriptionId = "SendComponent";
|
|
|
|
// FIXME(https://bitwarden.atlassian.net/browse/CL-764): Migrate to OnPush
|
|
// eslint-disable-next-line @angular-eslint/prefer-on-push-component-change-detection
|
|
@Component({
|
|
selector: "app-send",
|
|
templateUrl: "send.component.html",
|
|
imports: [CommonModule, JslibModule, FormsModule, NavComponent, AddEditComponent],
|
|
})
|
|
export class SendComponent extends BaseSendComponent implements OnInit, OnDestroy {
|
|
// FIXME(https://bitwarden.atlassian.net/browse/CL-903): Migrate to Signals
|
|
// eslint-disable-next-line @angular-eslint/prefer-signals
|
|
@ViewChild(AddEditComponent) addEditComponent: AddEditComponent;
|
|
|
|
sendId: string;
|
|
action: Action = Action.None;
|
|
|
|
constructor(
|
|
sendService: SendService,
|
|
i18nService: I18nService,
|
|
platformUtilsService: PlatformUtilsService,
|
|
environmentService: EnvironmentService,
|
|
private broadcasterService: BroadcasterService,
|
|
ngZone: NgZone,
|
|
searchService: SearchService,
|
|
policyService: PolicyService,
|
|
private searchBarService: SearchBarService,
|
|
logService: LogService,
|
|
sendApiService: SendApiService,
|
|
dialogService: DialogService,
|
|
toastService: ToastService,
|
|
accountService: AccountService,
|
|
) {
|
|
super(
|
|
sendService,
|
|
i18nService,
|
|
platformUtilsService,
|
|
environmentService,
|
|
ngZone,
|
|
searchService,
|
|
policyService,
|
|
logService,
|
|
sendApiService,
|
|
dialogService,
|
|
toastService,
|
|
accountService,
|
|
);
|
|
// eslint-disable-next-line rxjs-angular/prefer-takeuntil
|
|
this.searchBarService.searchText$.subscribe((searchText) => {
|
|
this.searchText = searchText;
|
|
this.searchTextChanged();
|
|
});
|
|
}
|
|
|
|
async ngOnInit() {
|
|
this.searchBarService.setEnabled(true);
|
|
this.searchBarService.setPlaceholderText(this.i18nService.t("searchSends"));
|
|
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
super.ngOnInit();
|
|
this.broadcasterService.subscribe(BroadcasterSubscriptionId, (message: any) => {
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
this.ngZone.run(async () => {
|
|
switch (message.command) {
|
|
case "syncCompleted":
|
|
await this.load();
|
|
break;
|
|
}
|
|
});
|
|
});
|
|
await this.load();
|
|
}
|
|
|
|
ngOnDestroy() {
|
|
this.broadcasterService.unsubscribe(BroadcasterSubscriptionId);
|
|
this.searchBarService.setEnabled(false);
|
|
}
|
|
|
|
async addSend() {
|
|
this.action = Action.Add;
|
|
if (this.addEditComponent != null) {
|
|
await this.addEditComponent.resetAndLoad();
|
|
}
|
|
}
|
|
|
|
cancel(s: SendView) {
|
|
this.action = Action.None;
|
|
this.sendId = null;
|
|
}
|
|
|
|
async deletedSend(s: SendView) {
|
|
await this.refresh();
|
|
this.action = Action.None;
|
|
this.sendId = null;
|
|
}
|
|
|
|
async savedSend(s: SendView) {
|
|
await this.refresh();
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
this.selectSend(s.id);
|
|
}
|
|
|
|
async selectSend(sendId: string) {
|
|
if (sendId === this.sendId && this.action === Action.Edit) {
|
|
return;
|
|
}
|
|
this.action = Action.Edit;
|
|
this.sendId = sendId;
|
|
if (this.addEditComponent != null) {
|
|
this.addEditComponent.sendId = sendId;
|
|
await this.addEditComponent.refresh();
|
|
}
|
|
}
|
|
|
|
get selectedSendType() {
|
|
return this.sends.find((s) => s.id === this.sendId)?.type;
|
|
}
|
|
|
|
viewSendMenu(send: SendView) {
|
|
const menu: RendererMenuItem[] = [];
|
|
menu.push({
|
|
label: this.i18nService.t("copyLink"),
|
|
click: () => this.copy(send),
|
|
});
|
|
if (send.password && !send.disabled) {
|
|
menu.push({
|
|
label: this.i18nService.t("removePassword"),
|
|
click: async () => {
|
|
await this.removePassword(send);
|
|
if (this.sendId === send.id) {
|
|
this.sendId = null;
|
|
// FIXME: Verify that this floating promise is intentional. If it is, add an explanatory comment and ensure there is proper error handling.
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
this.selectSend(send.id);
|
|
}
|
|
},
|
|
});
|
|
}
|
|
menu.push({
|
|
label: this.i18nService.t("delete"),
|
|
click: async () => {
|
|
await this.delete(send);
|
|
await this.deletedSend(send);
|
|
},
|
|
});
|
|
|
|
invokeMenu(menu);
|
|
}
|
|
}
|