mirror of
https://github.com/bitwarden/browser
synced 2025-12-17 08:43:33 +00:00
Password reprompt (#929)
* Use passwordRepromptService * Rename passwordPrompt to reprompt. Protect bulk actions * Change card to hidden, minor refactor. * Explicit reprompt value check * Ensure locales are the same on all platforms * Move showPasswordDialog to platformutils * Fix sweet alert validation message margin * Update locale to be the same as browser
This commit is contained in:
2
jslib
2
jslib
Submodule jslib updated: 4eb50d757d...a72c8a60c1
@@ -10,6 +10,7 @@ import { ApiService } from 'jslib/abstractions/api.service';
|
|||||||
import { CipherService } from 'jslib/abstractions/cipher.service';
|
import { CipherService } from 'jslib/abstractions/cipher.service';
|
||||||
import { EventService } from 'jslib/abstractions/event.service';
|
import { EventService } from 'jslib/abstractions/event.service';
|
||||||
import { I18nService } from 'jslib/abstractions/i18n.service';
|
import { I18nService } from 'jslib/abstractions/i18n.service';
|
||||||
|
import { PasswordRepromptService } from 'jslib/abstractions/passwordReprompt.service';
|
||||||
import { PlatformUtilsService } from 'jslib/abstractions/platformUtils.service';
|
import { PlatformUtilsService } from 'jslib/abstractions/platformUtils.service';
|
||||||
import { SearchService } from 'jslib/abstractions/search.service';
|
import { SearchService } from 'jslib/abstractions/search.service';
|
||||||
import { TotpService } from 'jslib/abstractions/totp.service';
|
import { TotpService } from 'jslib/abstractions/totp.service';
|
||||||
@@ -34,9 +35,10 @@ export class CiphersComponent extends BaseCiphersComponent {
|
|||||||
|
|
||||||
constructor(searchService: SearchService, toasterService: ToasterService, i18nService: I18nService,
|
constructor(searchService: SearchService, toasterService: ToasterService, i18nService: I18nService,
|
||||||
platformUtilsService: PlatformUtilsService, cipherService: CipherService,
|
platformUtilsService: PlatformUtilsService, cipherService: CipherService,
|
||||||
private apiService: ApiService, eventService: EventService, totpService: TotpService, userService: UserService) {
|
private apiService: ApiService, eventService: EventService, totpService: TotpService,
|
||||||
|
userService: UserService, passwordRepromptService: PasswordRepromptService) {
|
||||||
super(searchService, toasterService, i18nService, platformUtilsService, cipherService,
|
super(searchService, toasterService, i18nService, platformUtilsService, cipherService,
|
||||||
eventService, totpService, userService);
|
eventService, totpService, userService, passwordRepromptService);
|
||||||
}
|
}
|
||||||
|
|
||||||
async load(filter: (cipher: CipherView) => boolean = null) {
|
async load(filter: (cipher: CipherView) => boolean = null) {
|
||||||
|
|||||||
@@ -54,7 +54,6 @@ import { VaultTimeoutService } from 'jslib/services/vaultTimeout.service';
|
|||||||
import { WebCryptoFunctionService } from 'jslib/services/webCryptoFunction.service';
|
import { WebCryptoFunctionService } from 'jslib/services/webCryptoFunction.service';
|
||||||
|
|
||||||
import { ApiService as ApiServiceAbstraction } from 'jslib/abstractions/api.service';
|
import { ApiService as ApiServiceAbstraction } from 'jslib/abstractions/api.service';
|
||||||
import { AppIdService as AppIdServiceAbstraction } from 'jslib/abstractions/appId.service';
|
|
||||||
import { AuditService as AuditServiceAbstraction } from 'jslib/abstractions/audit.service';
|
import { AuditService as AuditServiceAbstraction } from 'jslib/abstractions/audit.service';
|
||||||
import { AuthService as AuthServiceAbstraction } from 'jslib/abstractions/auth.service';
|
import { AuthService as AuthServiceAbstraction } from 'jslib/abstractions/auth.service';
|
||||||
import { CipherService as CipherServiceAbstraction } from 'jslib/abstractions/cipher.service';
|
import { CipherService as CipherServiceAbstraction } from 'jslib/abstractions/cipher.service';
|
||||||
@@ -68,12 +67,12 @@ import { FileUploadService as FileUploadServiceAbstraction } from 'jslib/abstra
|
|||||||
import { FolderService as FolderServiceAbstraction } from 'jslib/abstractions/folder.service';
|
import { FolderService as FolderServiceAbstraction } from 'jslib/abstractions/folder.service';
|
||||||
import { I18nService as I18nServiceAbstraction } from 'jslib/abstractions/i18n.service';
|
import { I18nService as I18nServiceAbstraction } from 'jslib/abstractions/i18n.service';
|
||||||
import { ImportService as ImportServiceAbstraction } from 'jslib/abstractions/import.service';
|
import { ImportService as ImportServiceAbstraction } from 'jslib/abstractions/import.service';
|
||||||
import { LogService as LogServiceAbstraction } from 'jslib/abstractions/log.service';
|
|
||||||
import { MessagingService as MessagingServiceAbstraction } from 'jslib/abstractions/messaging.service';
|
import { MessagingService as MessagingServiceAbstraction } from 'jslib/abstractions/messaging.service';
|
||||||
import { NotificationsService as NotificationsServiceAbstraction } from 'jslib/abstractions/notifications.service';
|
import { NotificationsService as NotificationsServiceAbstraction } from 'jslib/abstractions/notifications.service';
|
||||||
import {
|
import {
|
||||||
PasswordGenerationService as PasswordGenerationServiceAbstraction,
|
PasswordGenerationService as PasswordGenerationServiceAbstraction,
|
||||||
} from 'jslib/abstractions/passwordGeneration.service';
|
} from 'jslib/abstractions/passwordGeneration.service';
|
||||||
|
import { PasswordRepromptService as PasswordRepromptServiceAbstraction } from 'jslib/abstractions/passwordReprompt.service';
|
||||||
import { PlatformUtilsService as PlatformUtilsServiceAbstraction } from 'jslib/abstractions/platformUtils.service';
|
import { PlatformUtilsService as PlatformUtilsServiceAbstraction } from 'jslib/abstractions/platformUtils.service';
|
||||||
import { PolicyService as PolicyServiceAbstraction } from 'jslib/abstractions/policy.service';
|
import { PolicyService as PolicyServiceAbstraction } from 'jslib/abstractions/policy.service';
|
||||||
import { SearchService as SearchServiceAbstraction } from 'jslib/abstractions/search.service';
|
import { SearchService as SearchServiceAbstraction } from 'jslib/abstractions/search.service';
|
||||||
@@ -86,6 +85,7 @@ import { TokenService as TokenServiceAbstraction } from 'jslib/abstractions/toke
|
|||||||
import { TotpService as TotpServiceAbstraction } from 'jslib/abstractions/totp.service';
|
import { TotpService as TotpServiceAbstraction } from 'jslib/abstractions/totp.service';
|
||||||
import { UserService as UserServiceAbstraction } from 'jslib/abstractions/user.service';
|
import { UserService as UserServiceAbstraction } from 'jslib/abstractions/user.service';
|
||||||
import { VaultTimeoutService as VaultTimeoutServiceAbstraction } from 'jslib/abstractions/vaultTimeout.service';
|
import { VaultTimeoutService as VaultTimeoutServiceAbstraction } from 'jslib/abstractions/vaultTimeout.service';
|
||||||
|
import { PasswordRepromptService } from 'jslib/services/passwordReprompt.service';
|
||||||
|
|
||||||
const i18nService = new I18nService(window.navigator.language, 'locales');
|
const i18nService = new I18nService(window.navigator.language, 'locales');
|
||||||
const stateService = new StateService();
|
const stateService = new StateService();
|
||||||
@@ -137,6 +137,7 @@ const notificationsService = new NotificationsService(userService, syncService,
|
|||||||
const environmentService = new EnvironmentService(apiService, storageService, notificationsService);
|
const environmentService = new EnvironmentService(apiService, storageService, notificationsService);
|
||||||
const auditService = new AuditService(cryptoFunctionService, apiService);
|
const auditService = new AuditService(cryptoFunctionService, apiService);
|
||||||
const eventLoggingService = new EventLoggingService(storageService, apiService, userService, cipherService);
|
const eventLoggingService = new EventLoggingService(storageService, apiService, userService, cipherService);
|
||||||
|
const passwordRepromptService = new PasswordRepromptService(i18nService, cryptoService, platformUtilsService);
|
||||||
|
|
||||||
containerService.attachToWindow(window);
|
containerService.attachToWindow(window);
|
||||||
|
|
||||||
@@ -222,6 +223,7 @@ export function initFactory(): Function {
|
|||||||
{ provide: EventLoggingServiceAbstraction, useValue: eventLoggingService },
|
{ provide: EventLoggingServiceAbstraction, useValue: eventLoggingService },
|
||||||
{ provide: PolicyServiceAbstraction, useValue: policyService },
|
{ provide: PolicyServiceAbstraction, useValue: policyService },
|
||||||
{ provide: SendServiceAbstraction, useValue: sendService },
|
{ provide: SendServiceAbstraction, useValue: sendService },
|
||||||
|
{ provide: PasswordRepromptServiceAbstraction, useValue: passwordRepromptService },
|
||||||
{
|
{
|
||||||
provide: APP_INITIALIZER,
|
provide: APP_INITIALIZER,
|
||||||
useFactory: initFactory,
|
useFactory: initFactory,
|
||||||
|
|||||||
@@ -209,10 +209,16 @@
|
|||||||
<div class="col-6 form-group">
|
<div class="col-6 form-group">
|
||||||
<label for="cardNumber">{{'number' | i18n}}</label>
|
<label for="cardNumber">{{'number' | i18n}}</label>
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<input id="cardNumber" class="form-control" type="text" name="Card.Number"
|
<input id="cardNumber" class="form-control text-monospace"
|
||||||
[(ngModel)]="cipher.card.number" appInputVerbatim
|
type="{{showCardNumber ? 'text' : 'password'}}" name="Card.Number"
|
||||||
|
[(ngModel)]="cipher.card.number" appInputVerbatim autocomplete="new-password"
|
||||||
[disabled]="cipher.isDeleted || viewOnly">
|
[disabled]="cipher.isDeleted || viewOnly">
|
||||||
<div class="input-group-append">
|
<div class="input-group-append">
|
||||||
|
<button type="button" class="btn btn-outline-secondary"
|
||||||
|
appA11yTitle="{{'toggleVisibility' | i18n}}" (click)="toggleCardNumber()">
|
||||||
|
<i class="fa fa-lg" aria-hidden="true"
|
||||||
|
[ngClass]="{'fa-eye': !showCardNumber, 'fa-eye-slash': showCardNumber}"></i>
|
||||||
|
</button>
|
||||||
<button type="button" class="btn btn-outline-secondary"
|
<button type="button" class="btn btn-outline-secondary"
|
||||||
appA11yTitle="{{'copyNumber' | i18n}}"
|
appA11yTitle="{{'copyNumber' | i18n}}"
|
||||||
(click)="copy(cipher.card.number, 'number', 'Number')">
|
(click)="copy(cipher.card.number, 'number', 'Number')">
|
||||||
@@ -512,6 +518,14 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
<ng-container>
|
||||||
|
<h3 class="mt-4">{{'options' | i18n}}</h3>
|
||||||
|
<div class="form-check">
|
||||||
|
<input class="form-check-input" type="checkbox" [ngModel]="reprompt" (change)="repromptChanged()"
|
||||||
|
id="passwordPrompt" name="passwordPrompt" [disabled]="cipher.isDeleted || viewOnly">
|
||||||
|
<label class="form-check-label" for="passwordPrompt">{{'passwordPrompt' | i18n}}</label>
|
||||||
|
</div>
|
||||||
|
</ng-container>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<button type="submit" class="btn btn-primary btn-submit" [disabled]="form.loading" *ngIf="!viewOnly">
|
<button type="submit" class="btn btn-primary btn-submit" [disabled]="form.loading" *ngIf="!viewOnly">
|
||||||
|
|||||||
@@ -8,6 +8,8 @@ import {
|
|||||||
import { ToasterService } from 'angular2-toaster';
|
import { ToasterService } from 'angular2-toaster';
|
||||||
|
|
||||||
import { I18nService } from 'jslib/abstractions/i18n.service';
|
import { I18nService } from 'jslib/abstractions/i18n.service';
|
||||||
|
import { PasswordRepromptService } from 'jslib/abstractions/passwordReprompt.service';
|
||||||
|
import { CipherRepromptType } from 'jslib/enums/cipherRepromptType';
|
||||||
|
|
||||||
import { Organization } from 'jslib/models/domain/organization';
|
import { Organization } from 'jslib/models/domain/organization';
|
||||||
|
|
||||||
@@ -36,9 +38,14 @@ export class BulkActionsComponent {
|
|||||||
|
|
||||||
constructor(private toasterService: ToasterService,
|
constructor(private toasterService: ToasterService,
|
||||||
private i18nService: I18nService,
|
private i18nService: I18nService,
|
||||||
private componentFactoryResolver: ComponentFactoryResolver) { }
|
private componentFactoryResolver: ComponentFactoryResolver,
|
||||||
|
private passwordRepromptService: PasswordRepromptService) { }
|
||||||
|
|
||||||
|
async bulkDelete() {
|
||||||
|
if (!await this.promptPassword()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
bulkDelete() {
|
|
||||||
const selectedIds = this.ciphersComponent.getSelectedIds();
|
const selectedIds = this.ciphersComponent.getSelectedIds();
|
||||||
if (selectedIds.length === 0) {
|
if (selectedIds.length === 0) {
|
||||||
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
||||||
@@ -67,7 +74,11 @@ export class BulkActionsComponent {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
bulkRestore() {
|
async bulkRestore() {
|
||||||
|
if (!await this.promptPassword()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const selectedIds = this.ciphersComponent.getSelectedIds();
|
const selectedIds = this.ciphersComponent.getSelectedIds();
|
||||||
if (selectedIds.length === 0) {
|
if (selectedIds.length === 0) {
|
||||||
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
||||||
@@ -94,7 +105,11 @@ export class BulkActionsComponent {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
bulkShare() {
|
async bulkShare() {
|
||||||
|
if (!await this.promptPassword()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const selectedCiphers = this.ciphersComponent.getSelected();
|
const selectedCiphers = this.ciphersComponent.getSelected();
|
||||||
if (selectedCiphers.length === 0) {
|
if (selectedCiphers.length === 0) {
|
||||||
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
||||||
@@ -121,7 +136,11 @@ export class BulkActionsComponent {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
bulkMove() {
|
async bulkMove() {
|
||||||
|
if (!await this.promptPassword()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const selectedIds = this.ciphersComponent.getSelectedIds();
|
const selectedIds = this.ciphersComponent.getSelectedIds();
|
||||||
if (selectedIds.length === 0) {
|
if (selectedIds.length === 0) {
|
||||||
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
this.toasterService.popAsync('error', this.i18nService.t('errorOccurred'),
|
||||||
@@ -151,4 +170,11 @@ export class BulkActionsComponent {
|
|||||||
selectAll(select: boolean) {
|
selectAll(select: boolean) {
|
||||||
this.ciphersComponent.selectAll(select);
|
this.ciphersComponent.selectAll(select);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async promptPassword() {
|
||||||
|
const selectedCiphers = this.ciphersComponent.getSelected();
|
||||||
|
const notProtected = !selectedCiphers.find(cipher => cipher.reprompt !== CipherRepromptType.None);
|
||||||
|
|
||||||
|
return notProtected || await this.passwordRepromptService.showPasswordPrompt();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,12 +38,12 @@
|
|||||||
<div class="dropdown-menu dropdown-menu-right" aria-labelledby="dropdownMenuButton">
|
<div class="dropdown-menu dropdown-menu-right" aria-labelledby="dropdownMenuButton">
|
||||||
<ng-container *ngIf="c.type === cipherType.Login && !c.isDeleted">
|
<ng-container *ngIf="c.type === cipherType.Login && !c.isDeleted">
|
||||||
<a class="dropdown-item" href="#" appStopClick
|
<a class="dropdown-item" href="#" appStopClick
|
||||||
(click)="copy(c, c.login.username, 'username', 'username')">
|
(click)="copy(c, c.login.username, 'username', 'Username')">
|
||||||
<i class="fa fa-fw fa-clone" aria-hidden="true"></i>
|
<i class="fa fa-fw fa-clone" aria-hidden="true"></i>
|
||||||
{{'copyUsername' | i18n}}
|
{{'copyUsername' | i18n}}
|
||||||
</a>
|
</a>
|
||||||
<a class="dropdown-item" href="#" appStopClick
|
<a class="dropdown-item" href="#" appStopClick
|
||||||
(click)="copy(c, c.login.password, 'password', 'password')" *ngIf="c.viewPassword">
|
(click)="copy(c, c.login.password, 'password', 'Password')" *ngIf="c.viewPassword">
|
||||||
<i class="fa fa-fw fa-clone" aria-hidden="true"></i>
|
<i class="fa fa-fw fa-clone" aria-hidden="true"></i>
|
||||||
{{'copyPassword' | i18n}}
|
{{'copyPassword' | i18n}}
|
||||||
</a>
|
</a>
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { ToasterService } from 'angular2-toaster';
|
|||||||
import { CipherService } from 'jslib/abstractions/cipher.service';
|
import { CipherService } from 'jslib/abstractions/cipher.service';
|
||||||
import { EventService } from 'jslib/abstractions/event.service';
|
import { EventService } from 'jslib/abstractions/event.service';
|
||||||
import { I18nService } from 'jslib/abstractions/i18n.service';
|
import { I18nService } from 'jslib/abstractions/i18n.service';
|
||||||
|
import { PasswordRepromptService } from 'jslib/abstractions/passwordReprompt.service';
|
||||||
import { PlatformUtilsService } from 'jslib/abstractions/platformUtils.service';
|
import { PlatformUtilsService } from 'jslib/abstractions/platformUtils.service';
|
||||||
import { SearchService } from 'jslib/abstractions/search.service';
|
import { SearchService } from 'jslib/abstractions/search.service';
|
||||||
import { TotpService } from 'jslib/abstractions/totp.service';
|
import { TotpService } from 'jslib/abstractions/totp.service';
|
||||||
@@ -18,6 +19,7 @@ import { UserService } from 'jslib/abstractions/user.service';
|
|||||||
|
|
||||||
import { CiphersComponent as BaseCiphersComponent } from 'jslib/angular/components/ciphers.component';
|
import { CiphersComponent as BaseCiphersComponent } from 'jslib/angular/components/ciphers.component';
|
||||||
|
|
||||||
|
import { CipherRepromptType } from 'jslib/enums/cipherRepromptType';
|
||||||
import { CipherType } from 'jslib/enums/cipherType';
|
import { CipherType } from 'jslib/enums/cipherType';
|
||||||
import { EventType } from 'jslib/enums/eventType';
|
import { EventType } from 'jslib/enums/eventType';
|
||||||
|
|
||||||
@@ -43,7 +45,8 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
constructor(searchService: SearchService, protected toasterService: ToasterService,
|
constructor(searchService: SearchService, protected toasterService: ToasterService,
|
||||||
protected i18nService: I18nService, protected platformUtilsService: PlatformUtilsService,
|
protected i18nService: I18nService, protected platformUtilsService: PlatformUtilsService,
|
||||||
protected cipherService: CipherService, protected eventService: EventService,
|
protected cipherService: CipherService, protected eventService: EventService,
|
||||||
protected totpService: TotpService, protected userService: UserService) {
|
protected totpService: TotpService, protected userService: UserService,
|
||||||
|
protected passwordRepromptService: PasswordRepromptService) {
|
||||||
super(searchService);
|
super(searchService);
|
||||||
this.pageSize = 200;
|
this.pageSize = 200;
|
||||||
}
|
}
|
||||||
@@ -60,11 +63,17 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
this.platformUtilsService.launchUri(uri);
|
this.platformUtilsService.launchUri(uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
attachments(c: CipherView) {
|
async attachments(c: CipherView) {
|
||||||
|
if (!await this.repromptCipher(c)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.onAttachmentsClicked.emit(c);
|
this.onAttachmentsClicked.emit(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
share(c: CipherView) {
|
async share(c: CipherView) {
|
||||||
|
if (!await this.repromptCipher(c)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.onShareClicked.emit(c);
|
this.onShareClicked.emit(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,11 +81,17 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
this.onCollectionsClicked.emit(c);
|
this.onCollectionsClicked.emit(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
clone(c: CipherView) {
|
async clone(c: CipherView) {
|
||||||
|
if (!await this.repromptCipher(c)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.onCloneClicked.emit(c);
|
this.onCloneClicked.emit(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
async delete(c: CipherView): Promise<boolean> {
|
async delete(c: CipherView): Promise<boolean> {
|
||||||
|
if (!await this.repromptCipher(c)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (this.actionPromise != null) {
|
if (this.actionPromise != null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -121,12 +136,20 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
}
|
}
|
||||||
|
|
||||||
async copy(cipher: CipherView, value: string, typeI18nKey: string, aType: string) {
|
async copy(cipher: CipherView, value: string, typeI18nKey: string, aType: string) {
|
||||||
|
if (this.passwordRepromptService.protectedFields().includes(aType) && !await this.repromptCipher(cipher)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (value == null || aType === 'TOTP' && !this.displayTotpCopyButton(cipher)) {
|
if (value == null || aType === 'TOTP' && !this.displayTotpCopyButton(cipher)) {
|
||||||
return;
|
return;
|
||||||
} else if (value === cipher.login.totp) {
|
} else if (value === cipher.login.totp) {
|
||||||
value = await this.totpService.getCode(value);
|
value = await this.totpService.getCode(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!cipher.viewPassword) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.platformUtilsService.copyToClipboard(value, { window: window });
|
this.platformUtilsService.copyToClipboard(value, { window: window });
|
||||||
this.toasterService.popAsync('info', null,
|
this.toasterService.popAsync('info', null,
|
||||||
this.i18nService.t('valueCopied', this.i18nService.t(typeI18nKey)));
|
this.i18nService.t('valueCopied', this.i18nService.t(typeI18nKey)));
|
||||||
@@ -170,6 +193,12 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
(cipher.organizationUseTotp || this.userHasPremiumAccess);
|
(cipher.organizationUseTotp || this.userHasPremiumAccess);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async selectCipher(cipher: CipherView) {
|
||||||
|
if (await this.repromptCipher(cipher)) {
|
||||||
|
super.selectCipher(cipher);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected deleteCipher(id: string, permanent: boolean) {
|
protected deleteCipher(id: string, permanent: boolean) {
|
||||||
return permanent ? this.cipherService.deleteWithServer(id) : this.cipherService.softDeleteWithServer(id);
|
return permanent ? this.cipherService.deleteWithServer(id) : this.cipherService.softDeleteWithServer(id);
|
||||||
}
|
}
|
||||||
@@ -177,4 +206,8 @@ export class CiphersComponent extends BaseCiphersComponent implements OnDestroy
|
|||||||
protected showFixOldAttachments(c: CipherView) {
|
protected showFixOldAttachments(c: CipherView) {
|
||||||
return c.hasOldAttachments && c.organizationId == null;
|
return c.hasOldAttachments && c.organizationId == null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected async repromptCipher(c: CipherView) {
|
||||||
|
return c.reprompt === CipherRepromptType.None || await this.passwordRepromptService.showPasswordPrompt();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3881,5 +3881,14 @@
|
|||||||
},
|
},
|
||||||
"trashCleanupWarningSelfHosted": {
|
"trashCleanupWarningSelfHosted": {
|
||||||
"message": "Ciphers that have been in Trash for a while will be automatically deleted."
|
"message": "Ciphers that have been in Trash for a while will be automatically deleted."
|
||||||
|
},
|
||||||
|
"passwordPrompt": {
|
||||||
|
"message": "Master password re-prompt"
|
||||||
|
},
|
||||||
|
"passwordConfirmation": {
|
||||||
|
"message": "Master password confirmation"
|
||||||
|
},
|
||||||
|
"passwordConfirmationDesc": {
|
||||||
|
"message": "This action is protected. To continue, please re-enter your master password to verify your identity."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -162,6 +162,10 @@ $fa-font-path: "~font-awesome/fonts";
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.swal2-validation-message {
|
||||||
|
margin: 0 -15px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
date-input-polyfill {
|
date-input-polyfill {
|
||||||
|
|||||||
@@ -211,6 +211,32 @@ export class WebPlatformUtilsService implements PlatformUtilsService {
|
|||||||
return confirmed.value;
|
return confirmed.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async showPasswordDialog(title: string, body: string, passwordValidation: (value: string) => Promise<boolean>):
|
||||||
|
Promise<boolean> {
|
||||||
|
const result = await Swal.fire({
|
||||||
|
heightAuto: false,
|
||||||
|
title: title,
|
||||||
|
input: 'password',
|
||||||
|
text: body,
|
||||||
|
confirmButtonText: this.i18nService.t('ok'),
|
||||||
|
showCancelButton: true,
|
||||||
|
cancelButtonText: this.i18nService.t('cancel'),
|
||||||
|
inputAttributes: {
|
||||||
|
autocapitalize: 'off',
|
||||||
|
autocorrect: 'off',
|
||||||
|
},
|
||||||
|
inputValidator: async (value: string): Promise<any> => {
|
||||||
|
if (await passwordValidation(value)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.i18nService.t('invalidMasterPassword');
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return result.isConfirmed;
|
||||||
|
}
|
||||||
|
|
||||||
isDev(): boolean {
|
isDev(): boolean {
|
||||||
return process.env.ENV === 'development';
|
return process.env.ENV === 'development';
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user