mirror of
https://github.com/bitwarden/web
synced 2025-12-06 00:03:28 +00:00
Merge branch 'master' of github.com:bitwarden/web into feature/cl-forms
# Conflicts: # jslib
This commit is contained in:
@@ -22,9 +22,9 @@ const routes: Routes = [
|
|||||||
component: ManageComponent,
|
component: ManageComponent,
|
||||||
canActivate: [PermissionsGuard],
|
canActivate: [PermissionsGuard],
|
||||||
data: {
|
data: {
|
||||||
permissions: [
|
permissions: NavigationPermissionsService.getPermissions("manage").concat(
|
||||||
NavigationPermissionsService.getPermissions("manage").concat(Permissions.ManageSso),
|
Permissions.ManageSso
|
||||||
],
|
),
|
||||||
},
|
},
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
|
|||||||
2
jslib
2
jslib
Submodule jslib updated: 8a0f096fd6...be870373d3
1300
package-lock.json
generated
1300
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -55,7 +55,7 @@ export class LockComponent extends BaseLockComponent {
|
|||||||
await super.ngOnInit();
|
await super.ngOnInit();
|
||||||
this.onSuccessfulSubmit = async () => {
|
this.onSuccessfulSubmit = async () => {
|
||||||
const previousUrl = this.routerService.getPreviousUrl();
|
const previousUrl = this.routerService.getPreviousUrl();
|
||||||
if (previousUrl !== "/" && previousUrl.indexOf("lock") === -1) {
|
if (previousUrl && previousUrl !== "/" && previousUrl.indexOf("lock") === -1) {
|
||||||
this.successRoute = previousUrl;
|
this.successRoute = previousUrl;
|
||||||
}
|
}
|
||||||
this.router.navigateByUrl(this.successRoute);
|
this.router.navigateByUrl(this.successRoute);
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ export class LoginComponent extends BaseLoginComponent {
|
|||||||
if (qParams.premium != null) {
|
if (qParams.premium != null) {
|
||||||
this.routerService.setPreviousUrl("/settings/premium");
|
this.routerService.setPreviousUrl("/settings/premium");
|
||||||
} else if (qParams.org != null) {
|
} else if (qParams.org != null) {
|
||||||
const route = this.router.createUrlTree(["settings/create-organization"], {
|
const route = this.router.createUrlTree(["create-organization"], {
|
||||||
queryParams: { plan: qParams.org },
|
queryParams: { plan: qParams.org },
|
||||||
});
|
});
|
||||||
this.routerService.setPreviousUrl(route.toString());
|
this.routerService.setPreviousUrl(route.toString());
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ export class RegisterComponent extends BaseRegisterComponent {
|
|||||||
} else if (qParams.org != null) {
|
} else if (qParams.org != null) {
|
||||||
this.showCreateOrgMessage = true;
|
this.showCreateOrgMessage = true;
|
||||||
this.referenceData.flow = qParams.org;
|
this.referenceData.flow = qParams.org;
|
||||||
const route = this.router.createUrlTree(["settings/create-organization"], {
|
const route = this.router.createUrlTree(["create-organization"], {
|
||||||
queryParams: { plan: qParams.org },
|
queryParams: { plan: qParams.org },
|
||||||
});
|
});
|
||||||
this.routerService.setPreviousUrl(route.toString());
|
this.routerService.setPreviousUrl(route.toString());
|
||||||
|
|||||||
@@ -58,7 +58,7 @@
|
|||||||
</li>
|
</li>
|
||||||
<bit-menu-divider></bit-menu-divider>
|
<bit-menu-divider></bit-menu-divider>
|
||||||
<li class="tw-list-none" role="none">
|
<li class="tw-list-none" role="none">
|
||||||
<a bit-menu-item routerLink="/settings/create-organization">
|
<a bit-menu-item routerLink="/create-organization">
|
||||||
<i class="bwi bwi-plus mr-2"></i>
|
<i class="bwi bwi-plus mr-2"></i>
|
||||||
{{ "newOrganization" | i18n }}</a
|
{{ "newOrganization" | i18n }}</a
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -38,7 +38,7 @@
|
|||||||
<li>
|
<li>
|
||||||
<button
|
<button
|
||||||
[bitMenuTriggerFor]="accountMenu"
|
[bitMenuTriggerFor]="accountMenu"
|
||||||
class="tw-border-0 tw-bg-transparent tw-text-contrast tw-opacity-70 hover:tw-opacity-90"
|
class="tw-border-0 tw-bg-transparent tw-text-alt2 tw-opacity-70 hover:tw-opacity-90"
|
||||||
>
|
>
|
||||||
<i class="bwi bwi-user-circle bwi-lg" aria-hidden="true"></i>
|
<i class="bwi bwi-user-circle bwi-lg" aria-hidden="true"></i>
|
||||||
<i class="bwi bwi-caret-down bwi-sm" aria-hidden="true"></i>
|
<i class="bwi bwi-caret-down bwi-sm" aria-hidden="true"></i>
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { Component, OnInit } from "@angular/core";
|
import { Component, NgZone, OnInit } from "@angular/core";
|
||||||
|
|
||||||
|
import { BroadcasterService } from "jslib-common/abstractions/broadcaster.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
||||||
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
||||||
@@ -31,7 +32,9 @@ export class NavbarComponent implements OnInit {
|
|||||||
private providerService: ProviderService,
|
private providerService: ProviderService,
|
||||||
private syncService: SyncService,
|
private syncService: SyncService,
|
||||||
private organizationService: OrganizationService,
|
private organizationService: OrganizationService,
|
||||||
private i18nService: I18nService
|
private i18nService: I18nService,
|
||||||
|
private broadcasterService: BroadcasterService,
|
||||||
|
private ngZone: NgZone
|
||||||
) {
|
) {
|
||||||
this.selfHosted = this.platformUtilsService.isSelfHost();
|
this.selfHosted = this.platformUtilsService.isSelfHost();
|
||||||
}
|
}
|
||||||
@@ -49,8 +52,24 @@ export class NavbarComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
this.providers = await this.providerService.getAll();
|
this.providers = await this.providerService.getAll();
|
||||||
|
|
||||||
|
this.organizations = await this.buildOrganizations();
|
||||||
|
|
||||||
|
this.broadcasterService.subscribe(this.constructor.name, async (message: any) => {
|
||||||
|
this.ngZone.run(async () => {
|
||||||
|
switch (message.command) {
|
||||||
|
case "organizationCreated":
|
||||||
|
if (this.organizations.length < 1) {
|
||||||
|
this.organizations = await this.buildOrganizations();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async buildOrganizations() {
|
||||||
const allOrgs = await this.organizationService.getAll();
|
const allOrgs = await this.organizationService.getAll();
|
||||||
this.organizations = allOrgs
|
return allOrgs
|
||||||
.filter((org) => OrgNavigationPermissionsService.canAccessAdmin(org))
|
.filter((org) => OrgNavigationPermissionsService.canAccessAdmin(org))
|
||||||
.sort(Utils.getSortFunction(this.i18nService, "name"));
|
.sort(Utils.getSortFunction(this.i18nService, "name"));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,7 +35,6 @@ import { BulkStatusComponent as OrgBulkStatusComponent } from "../organizations/
|
|||||||
import { CollectionAddEditComponent as OrgCollectionAddEditComponent } from "../organizations/manage/collection-add-edit.component";
|
import { CollectionAddEditComponent as OrgCollectionAddEditComponent } from "../organizations/manage/collection-add-edit.component";
|
||||||
import { CollectionsComponent as OrgManageCollectionsComponent } from "../organizations/manage/collections.component";
|
import { CollectionsComponent as OrgManageCollectionsComponent } from "../organizations/manage/collections.component";
|
||||||
import { EntityEventsComponent as OrgEntityEventsComponent } from "../organizations/manage/entity-events.component";
|
import { EntityEventsComponent as OrgEntityEventsComponent } from "../organizations/manage/entity-events.component";
|
||||||
import { EntityUsersComponent as OrgEntityUsersComponent } from "../organizations/manage/entity-users.component";
|
|
||||||
import { EventsComponent as OrgEventsComponent } from "../organizations/manage/events.component";
|
import { EventsComponent as OrgEventsComponent } from "../organizations/manage/events.component";
|
||||||
import { GroupAddEditComponent as OrgGroupAddEditComponent } from "../organizations/manage/group-add-edit.component";
|
import { GroupAddEditComponent as OrgGroupAddEditComponent } from "../organizations/manage/group-add-edit.component";
|
||||||
import { GroupsComponent as OrgGroupsComponent } from "../organizations/manage/groups.component";
|
import { GroupsComponent as OrgGroupsComponent } from "../organizations/manage/groups.component";
|
||||||
@@ -58,6 +57,7 @@ import { SingleOrgPolicyComponent } from "../organizations/policies/single-org.c
|
|||||||
import { TwoFactorAuthenticationPolicyComponent } from "../organizations/policies/two-factor-authentication.component";
|
import { TwoFactorAuthenticationPolicyComponent } from "../organizations/policies/two-factor-authentication.component";
|
||||||
import { AccountComponent as OrgAccountComponent } from "../organizations/settings/account.component";
|
import { AccountComponent as OrgAccountComponent } from "../organizations/settings/account.component";
|
||||||
import { AdjustSubscription } from "../organizations/settings/adjust-subscription.component";
|
import { AdjustSubscription } from "../organizations/settings/adjust-subscription.component";
|
||||||
|
import { BillingSyncApiKeyComponent } from "../organizations/settings/billing-sync-api-key.component";
|
||||||
import { ChangePlanComponent } from "../organizations/settings/change-plan.component";
|
import { ChangePlanComponent } from "../organizations/settings/change-plan.component";
|
||||||
import { DeleteOrganizationComponent } from "../organizations/settings/delete-organization.component";
|
import { DeleteOrganizationComponent } from "../organizations/settings/delete-organization.component";
|
||||||
import { DownloadLicenseComponent } from "../organizations/settings/download-license.component";
|
import { DownloadLicenseComponent } from "../organizations/settings/download-license.component";
|
||||||
@@ -66,6 +66,7 @@ import { OrganizationBillingComponent } from "../organizations/settings/organiza
|
|||||||
import { OrganizationSubscriptionComponent } from "../organizations/settings/organization-subscription.component";
|
import { OrganizationSubscriptionComponent } from "../organizations/settings/organization-subscription.component";
|
||||||
import { SettingsComponent as OrgSettingComponent } from "../organizations/settings/settings.component";
|
import { SettingsComponent as OrgSettingComponent } from "../organizations/settings/settings.component";
|
||||||
import { TwoFactorSetupComponent as OrgTwoFactorSetupComponent } from "../organizations/settings/two-factor-setup.component";
|
import { TwoFactorSetupComponent as OrgTwoFactorSetupComponent } from "../organizations/settings/two-factor-setup.component";
|
||||||
|
import { AcceptFamilySponsorshipComponent } from "../organizations/sponsorships/accept-family-sponsorship.component";
|
||||||
import { FamiliesForEnterpriseSetupComponent } from "../organizations/sponsorships/families-for-enterprise-setup.component";
|
import { FamiliesForEnterpriseSetupComponent } from "../organizations/sponsorships/families-for-enterprise-setup.component";
|
||||||
import { ExportComponent as OrgExportComponent } from "../organizations/tools/export.component";
|
import { ExportComponent as OrgExportComponent } from "../organizations/tools/export.component";
|
||||||
import { ExposedPasswordsReportComponent as OrgExposedPasswordsReportComponent } from "../organizations/tools/exposed-passwords-report.component";
|
import { ExposedPasswordsReportComponent as OrgExposedPasswordsReportComponent } from "../organizations/tools/exposed-passwords-report.component";
|
||||||
@@ -98,6 +99,7 @@ import { AddCreditComponent } from "../settings/add-credit.component";
|
|||||||
import { AdjustPaymentComponent } from "../settings/adjust-payment.component";
|
import { AdjustPaymentComponent } from "../settings/adjust-payment.component";
|
||||||
import { AdjustStorageComponent } from "../settings/adjust-storage.component";
|
import { AdjustStorageComponent } from "../settings/adjust-storage.component";
|
||||||
import { ApiKeyComponent } from "../settings/api-key.component";
|
import { ApiKeyComponent } from "../settings/api-key.component";
|
||||||
|
import { BillingSyncKeyComponent } from "../settings/billing-sync-key.component";
|
||||||
import { ChangeEmailComponent } from "../settings/change-email.component";
|
import { ChangeEmailComponent } from "../settings/change-email.component";
|
||||||
import { ChangeKdfComponent } from "../settings/change-kdf.component";
|
import { ChangeKdfComponent } from "../settings/change-kdf.component";
|
||||||
import { ChangePasswordComponent } from "../settings/change-password.component";
|
import { ChangePasswordComponent } from "../settings/change-password.component";
|
||||||
@@ -171,6 +173,7 @@ import { OrganizationBadgeModule } from "./vault/modules/organization-badge/orga
|
|||||||
declarations: [
|
declarations: [
|
||||||
PremiumBadgeComponent,
|
PremiumBadgeComponent,
|
||||||
AcceptEmergencyComponent,
|
AcceptEmergencyComponent,
|
||||||
|
AcceptFamilySponsorshipComponent,
|
||||||
AcceptOrganizationComponent,
|
AcceptOrganizationComponent,
|
||||||
AccessComponent,
|
AccessComponent,
|
||||||
AccountComponent,
|
AccountComponent,
|
||||||
@@ -183,6 +186,8 @@ import { OrganizationBadgeModule } from "./vault/modules/organization-badge/orga
|
|||||||
AdjustSubscription,
|
AdjustSubscription,
|
||||||
ApiKeyComponent,
|
ApiKeyComponent,
|
||||||
AttachmentsComponent,
|
AttachmentsComponent,
|
||||||
|
BillingSyncApiKeyComponent,
|
||||||
|
BillingSyncKeyComponent,
|
||||||
BreachReportComponent,
|
BreachReportComponent,
|
||||||
BulkActionsComponent,
|
BulkActionsComponent,
|
||||||
BulkDeleteComponent,
|
BulkDeleteComponent,
|
||||||
@@ -239,7 +244,6 @@ import { OrganizationBadgeModule } from "./vault/modules/organization-badge/orga
|
|||||||
OrgCollectionAddEditComponent,
|
OrgCollectionAddEditComponent,
|
||||||
OrgCollectionsComponent,
|
OrgCollectionsComponent,
|
||||||
OrgEntityEventsComponent,
|
OrgEntityEventsComponent,
|
||||||
OrgEntityUsersComponent,
|
|
||||||
OrgEventsComponent,
|
OrgEventsComponent,
|
||||||
OrgExportComponent,
|
OrgExportComponent,
|
||||||
OrgExposedPasswordsReportComponent,
|
OrgExposedPasswordsReportComponent,
|
||||||
@@ -400,7 +404,6 @@ import { OrganizationBadgeModule } from "./vault/modules/organization-badge/orga
|
|||||||
OrgCollectionAddEditComponent,
|
OrgCollectionAddEditComponent,
|
||||||
OrgCollectionsComponent,
|
OrgCollectionsComponent,
|
||||||
OrgEntityEventsComponent,
|
OrgEntityEventsComponent,
|
||||||
OrgEntityUsersComponent,
|
|
||||||
OrgEventsComponent,
|
OrgEventsComponent,
|
||||||
OrgExportComponent,
|
OrgExportComponent,
|
||||||
OrgExposedPasswordsReportComponent,
|
OrgExposedPasswordsReportComponent,
|
||||||
|
|||||||
@@ -29,52 +29,52 @@
|
|||||||
></i>
|
></i>
|
||||||
<span class="sr-only">{{ "loading" | i18n }}</span>
|
<span class="sr-only">{{ "loading" | i18n }}</span>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<cdk-virtual-scroll-viewport
|
||||||
class="modal-body"
|
itemSize="46"
|
||||||
*ngIf="
|
minBufferPx="600"
|
||||||
!loading && users && (users | search: searchText:'name':'email':'id') as searchedUsers
|
maxBufferPx="1200"
|
||||||
"
|
[style]="scrollViewportStyle"
|
||||||
>
|
>
|
||||||
<div class="d-flex">
|
<div class="modal-body" *ngIf="!loading && users && searchedUsers">
|
||||||
<div class="mr-3">
|
<div class="d-flex">
|
||||||
<label class="sr-only" for="search">{{ "search" | i18n }}</label>
|
<div class="mr-3">
|
||||||
<input
|
<label class="sr-only" for="search">{{ "search" | i18n }}</label>
|
||||||
type="search"
|
<input
|
||||||
class="form-control form-control-sm"
|
type="search"
|
||||||
id="search"
|
class="form-control form-control-sm"
|
||||||
placeholder="{{ 'search' | i18n }}"
|
id="search"
|
||||||
name="SearchText"
|
placeholder="{{ 'search' | i18n }}"
|
||||||
[(ngModel)]="searchText"
|
name="SearchText"
|
||||||
/>
|
[(ngModel)]="searchText"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="btn-group btn-group-sm" role="group">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
[ngClass]="{ active: !showSelected }"
|
||||||
|
(click)="filterSelected(false)"
|
||||||
|
>
|
||||||
|
{{ "all" | i18n }}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
[ngClass]="{ active: showSelected }"
|
||||||
|
(click)="filterSelected(true)"
|
||||||
|
>
|
||||||
|
{{ "selected" | i18n }}
|
||||||
|
<span class="badge badge-pill badge-info" *ngIf="selectedCount">{{
|
||||||
|
selectedCount
|
||||||
|
}}</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="btn-group btn-group-sm" role="group">
|
<ng-container *ngIf="!searchedUsers.length">
|
||||||
<button
|
<hr />
|
||||||
type="button"
|
{{ "noUsersInList" | i18n }}
|
||||||
class="btn btn-outline-secondary"
|
</ng-container>
|
||||||
[ngClass]="{ active: !showSelected }"
|
<table class="table table-hover table-list mb-0" [hidden]="!searchedUsers.length">
|
||||||
(click)="filterSelected(false)"
|
|
||||||
>
|
|
||||||
{{ "all" | i18n }}
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-outline-secondary"
|
|
||||||
[ngClass]="{ active: showSelected }"
|
|
||||||
(click)="filterSelected(true)"
|
|
||||||
>
|
|
||||||
{{ "selected" | i18n }}
|
|
||||||
<span class="badge badge-pill badge-info" *ngIf="selectedCount">{{
|
|
||||||
selectedCount
|
|
||||||
}}</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<ng-container *ngIf="!searchedUsers.length">
|
|
||||||
<hr />
|
|
||||||
{{ "noUsersInList" | i18n }}
|
|
||||||
</ng-container>
|
|
||||||
<ng-container *ngIf="searchedUsers.length">
|
|
||||||
<table class="table table-hover table-list mb-0">
|
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th> </th>
|
<th> </th>
|
||||||
@@ -91,7 +91,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr *ngFor="let u of searchedUsers">
|
<tr *cdkVirtualFor="let u of searchedUsers" class="">
|
||||||
<td class="table-list-checkbox" (click)="check(u)">
|
<td class="table-list-checkbox" (click)="check(u)">
|
||||||
<input
|
<input
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
@@ -164,8 +164,8 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</ng-container>
|
</div>
|
||||||
</div>
|
</cdk-virtual-scroll-viewport>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<button type="submit" class="btn btn-primary btn-submit" [disabled]="form.loading">
|
<button type="submit" class="btn btn-primary btn-submit" [disabled]="form.loading">
|
||||||
<i class="bwi bwi-spinner bwi-spin" title="{{ 'loading' | i18n }}" aria-hidden="true"></i>
|
<i class="bwi bwi-spinner bwi-spin" title="{{ 'loading' | i18n }}" aria-hidden="true"></i>
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
import { Component, EventEmitter, Input, OnInit, Output } from "@angular/core";
|
import { Component, EventEmitter, Input, OnInit, Output } from "@angular/core";
|
||||||
|
|
||||||
|
import { SearchPipe } from "jslib-angular/pipes/search.pipe";
|
||||||
import { ApiService } from "jslib-common/abstractions/api.service";
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { LogService } from "jslib-common/abstractions/log.service";
|
import { LogService } from "jslib-common/abstractions/log.service";
|
||||||
@@ -13,6 +14,7 @@ import { OrganizationUserUserDetailsResponse } from "jslib-common/models/respons
|
|||||||
@Component({
|
@Component({
|
||||||
selector: "app-entity-users",
|
selector: "app-entity-users",
|
||||||
templateUrl: "entity-users.component.html",
|
templateUrl: "entity-users.component.html",
|
||||||
|
providers: [SearchPipe],
|
||||||
})
|
})
|
||||||
export class EntityUsersComponent implements OnInit {
|
export class EntityUsersComponent implements OnInit {
|
||||||
@Input() entity: "group" | "collection";
|
@Input() entity: "group" | "collection";
|
||||||
@@ -33,6 +35,7 @@ export class EntityUsersComponent implements OnInit {
|
|||||||
private allUsers: OrganizationUserUserDetailsResponse[] = [];
|
private allUsers: OrganizationUserUserDetailsResponse[] = [];
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
private search: SearchPipe,
|
||||||
private apiService: ApiService,
|
private apiService: ApiService,
|
||||||
private i18nService: I18nService,
|
private i18nService: I18nService,
|
||||||
private platformUtilsService: PlatformUtilsService,
|
private platformUtilsService: PlatformUtilsService,
|
||||||
@@ -52,6 +55,14 @@ export class EntityUsersComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get searchedUsers() {
|
||||||
|
return this.search.transform(this.users, this.searchText, "name", "email", "id");
|
||||||
|
}
|
||||||
|
|
||||||
|
get scrollViewportStyle() {
|
||||||
|
return `min-height: 120px; height: ${120 + this.searchedUsers.length * 46}px`;
|
||||||
|
}
|
||||||
|
|
||||||
async loadUsers() {
|
async loadUsers() {
|
||||||
const users = await this.apiService.getOrganizationUsers(this.organizationId);
|
const users = await this.apiService.getOrganizationUsers(this.organizationId);
|
||||||
this.allUsers = users.data.map((r) => r).sort(Utils.getSortFunction(this.i18nService, "email"));
|
this.allUsers = users.data.map((r) => r).sort(Utils.getSortFunction(this.i18nService, "email"));
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
import { ScrollingModule } from "@angular/cdk/scrolling";
|
||||||
|
import { NgModule } from "@angular/core";
|
||||||
|
|
||||||
|
import { SharedModule } from "../../shared.module";
|
||||||
|
|
||||||
|
import { EntityUsersComponent } from "./entity-users.component";
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
imports: [SharedModule, ScrollingModule],
|
||||||
|
declarations: [EntityUsersComponent],
|
||||||
|
exports: [EntityUsersComponent],
|
||||||
|
})
|
||||||
|
export class OrganizationManageModule {}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
<ng-container *ngIf="!hide && !activeFilter.selectedOrganizationId">
|
<ng-container *ngIf="!hide">
|
||||||
<div class="filter-heading">
|
<div class="filter-heading">
|
||||||
<button
|
<button
|
||||||
class="toggle-button"
|
class="toggle-button"
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
</li>
|
</li>
|
||||||
<li class="filter-option">
|
<li class="filter-option">
|
||||||
<span class="filter-buttons">
|
<span class="filter-buttons">
|
||||||
<a href="#" routerLink="/settings/create-organization" class="filter-button">
|
<a href="#" routerLink="/create-organization" class="filter-button">
|
||||||
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
||||||
{{ "newOrganization" | i18n }}
|
{{ "newOrganization" | i18n }}
|
||||||
</a>
|
</a>
|
||||||
@@ -47,9 +47,9 @@
|
|||||||
</button>
|
</button>
|
||||||
<a
|
<a
|
||||||
href="#"
|
href="#"
|
||||||
routerLink="/settings/create-organization"
|
routerLink="/create-organization"
|
||||||
class="text-muted ml-auto create-organization-link"
|
class="text-muted ml-auto create-organization-link"
|
||||||
appA11yTitle="{{ 'addOrganization' | i18n }}"
|
appA11yTitle="{{ 'newOrganization' | i18n }}"
|
||||||
>
|
>
|
||||||
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
||||||
</a>
|
</a>
|
||||||
@@ -78,16 +78,14 @@
|
|||||||
</ul>
|
</ul>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-container *ngSwitchCase="'singleOrganizationAndPersonalOwnershipPolicies'">
|
<ng-container *ngSwitchCase="'singleOrganizationAndPersonalOwnershipPolicies'">
|
||||||
<ul class="filter-options">
|
<div class="filter-heading">
|
||||||
<li class="filter-option active">
|
<button class="filter-button active">
|
||||||
<button class="filter-button">
|
<i class="bwi bwi-fw bwi-business" aria-hidden="true"></i>
|
||||||
<i class="bwi bwi-fw bwi-business" aria-hidden="true"></i>
|
{{ organizations[0].name }}
|
||||||
{{ organizations[0].name }}
|
</button>
|
||||||
</button>
|
</div>
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-container *ngSwitchCase="'organizationMember'">
|
<ng-container *ngSwitchDefault>
|
||||||
<div class="filter-heading">
|
<div class="filter-heading">
|
||||||
<button
|
<button
|
||||||
class="toggle-button"
|
class="toggle-button"
|
||||||
@@ -114,9 +112,10 @@
|
|||||||
</button>
|
</button>
|
||||||
<a
|
<a
|
||||||
href="#"
|
href="#"
|
||||||
routerLink="/settings/create-organization"
|
routerLink="/create-organization"
|
||||||
class="text-muted ml-auto create-organization-link"
|
class="text-muted ml-auto create-organization-link"
|
||||||
appA11yTitle="{{ 'addOrganization' | i18n }}"
|
appA11yTitle="{{ 'newOrganization' | i18n }}"
|
||||||
|
*ngIf="!(displayMode === 'singleOrganizationPolicy')"
|
||||||
>
|
>
|
||||||
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>
|
||||||
</a>
|
</a>
|
||||||
@@ -140,7 +139,7 @@
|
|||||||
<i class="bwi bwi-fw bwi-business" aria-hidden="true"></i>
|
<i class="bwi bwi-fw bwi-business" aria-hidden="true"></i>
|
||||||
{{ organization.name }}
|
{{ organization.name }}
|
||||||
</button>
|
</button>
|
||||||
<ng-container *ngIf="organization.id === activeFilter.selectedOrganizationId">
|
<ng-container>
|
||||||
<button [bitMenuTriggerFor]="orgMenu" class="org-options ml-auto">
|
<button [bitMenuTriggerFor]="orgMenu" class="org-options ml-auto">
|
||||||
<i class="bwi bwi-ellipsis-v" aria-hidden="true"></i>
|
<i class="bwi bwi-ellipsis-v" aria-hidden="true"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|||||||
@@ -35,7 +35,6 @@ export class OrganizationOptionsComponent {
|
|||||||
) {}
|
) {}
|
||||||
|
|
||||||
async ngOnInit() {
|
async ngOnInit() {
|
||||||
await this.syncService.fullSync(true);
|
|
||||||
await this.load();
|
await this.load();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,6 +82,7 @@ export class OrganizationOptionsComponent {
|
|||||||
this.platformUtilsService.showToast("success", null, "Unlinked SSO");
|
this.platformUtilsService.showToast("success", null, "Unlinked SSO");
|
||||||
await this.load();
|
await this.load();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
this.platformUtilsService.showToast("error", this.i18nService.t("errorOccurred"), e.message);
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -107,6 +107,7 @@ export class OrganizationOptionsComponent {
|
|||||||
this.platformUtilsService.showToast("success", null, this.i18nService.t("leftOrganization"));
|
this.platformUtilsService.showToast("success", null, this.i18nService.t("leftOrganization"));
|
||||||
await this.load();
|
await this.load();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
this.platformUtilsService.showToast("error", this.i18nService.t("errorOccurred"), e.message);
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -174,6 +175,7 @@ export class OrganizationOptionsComponent {
|
|||||||
this.platformUtilsService.showToast("success", null, this.i18nService.t(toastStringRef));
|
this.platformUtilsService.showToast("success", null, this.i18nService.t(toastStringRef));
|
||||||
await this.load();
|
await this.load();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
this.platformUtilsService.showToast("error", this.i18nService.t("errorOccurred"), e.message);
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,19 +26,17 @@
|
|||||||
autocomplete="off"
|
autocomplete="off"
|
||||||
appAutofocus
|
appAutofocus
|
||||||
/>
|
/>
|
||||||
<div class="filter">
|
<app-organization-filter
|
||||||
<app-organization-filter
|
*ngIf="showOrgFilter"
|
||||||
*ngIf="showOrgFilter"
|
[hide]="hideOrganizations"
|
||||||
[hide]="hideOrganizations"
|
[activeFilter]="activeFilter"
|
||||||
[activeFilter]="activeFilter"
|
[collapsedFilterNodes]="collapsedFilterNodes"
|
||||||
[collapsedFilterNodes]="collapsedFilterNodes"
|
[organizations]="organizations"
|
||||||
[organizations]="organizations"
|
[activePersonalOwnershipPolicy]="activePersonalOwnershipPolicy"
|
||||||
[activePersonalOwnershipPolicy]="activePersonalOwnershipPolicy"
|
[activeSingleOrganizationPolicy]="activeSingleOrganizationPolicy"
|
||||||
[activeSingleOrganizationPolicy]="activeSingleOrganizationPolicy"
|
(onNodeCollapseStateChange)="toggleFilterNodeCollapseState($event)"
|
||||||
(onNodeCollapseStateChange)="toggleFilterNodeCollapseState($event)"
|
(onFilterChange)="applyFilter($event)"
|
||||||
(onFilterChange)="applyFilter($event)"
|
></app-organization-filter>
|
||||||
></app-organization-filter>
|
|
||||||
</div>
|
|
||||||
<div class="filter">
|
<div class="filter">
|
||||||
<app-status-filter
|
<app-status-filter
|
||||||
[hideFavorites]="!showFavorites"
|
[hideFavorites]="!showFavorites"
|
||||||
|
|||||||
@@ -32,6 +32,10 @@ export class VaultFilterComponent extends BaseVaultFilterComponent {
|
|||||||
// It should be removed as soon as doing so makes sense.
|
// It should be removed as soon as doing so makes sense.
|
||||||
async reloadOrganizations() {
|
async reloadOrganizations() {
|
||||||
this.organizations = await this.vaultFilterService.buildOrganizations();
|
this.organizations = await this.vaultFilterService.buildOrganizations();
|
||||||
|
this.activePersonalOwnershipPolicy =
|
||||||
|
await this.vaultFilterService.checkForPersonalOwnershipPolicy();
|
||||||
|
this.activeSingleOrganizationPolicy =
|
||||||
|
await this.vaultFilterService.checkForSingleOrganizationPolicy();
|
||||||
}
|
}
|
||||||
|
|
||||||
async initCollections() {
|
async initCollections() {
|
||||||
|
|||||||
@@ -32,19 +32,26 @@
|
|||||||
</small>
|
</small>
|
||||||
</h1>
|
</h1>
|
||||||
<div class="ml-auto d-flex">
|
<div class="ml-auto d-flex">
|
||||||
<app-vault-bulk-actions [ciphersComponent]="ciphersComponent" [deleted]="deleted">
|
<app-vault-bulk-actions
|
||||||
|
[ciphersComponent]="ciphersComponent"
|
||||||
|
[deleted]="activeFilter.status === 'trash'"
|
||||||
|
>
|
||||||
</app-vault-bulk-actions>
|
</app-vault-bulk-actions>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="btn btn-outline-primary btn-sm"
|
class="btn btn-outline-primary btn-sm"
|
||||||
(click)="addCipher()"
|
(click)="addCipher()"
|
||||||
*ngIf="!deleted"
|
*ngIf="activeFilter.status !== 'trash'"
|
||||||
>
|
>
|
||||||
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>{{ "addItem" | i18n }}
|
<i class="bwi bwi-plus bwi-fw" aria-hidden="true"></i>{{ "addItem" | i18n }}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<app-callout type="warning" *ngIf="deleted" icon="bwi-exclamation-triangle">
|
<app-callout
|
||||||
|
type="warning"
|
||||||
|
*ngIf="activeFilter.status === 'trash'"
|
||||||
|
icon="bwi-exclamation-triangle"
|
||||||
|
>
|
||||||
{{ trashCleanupWarning }}
|
{{ trashCleanupWarning }}
|
||||||
</app-callout>
|
</app-callout>
|
||||||
<app-vault-ciphers
|
<app-vault-ciphers
|
||||||
@@ -95,7 +102,10 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<p>{{ "premiumUpgradeUnlockFeatures" | i18n }}</p>
|
<p>{{ "premiumUpgradeUnlockFeatures" | i18n }}</p>
|
||||||
<a class="btn btn-block btn-outline-secondary" routerLink="/settings/premium">
|
<a
|
||||||
|
class="btn btn-block btn-outline-secondary"
|
||||||
|
routerLink="/settings/subscription/premium"
|
||||||
|
>
|
||||||
{{ "goPremium" | i18n }}
|
{{ "goPremium" | i18n }}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -209,7 +209,7 @@ export class IndividualVaultComponent implements OnInit, OnDestroy {
|
|||||||
cipherPassesFilter = cipher.type === this.activeFilter.cipherType;
|
cipherPassesFilter = cipher.type === this.activeFilter.cipherType;
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
this.activeFilter.selectedFolderId != null &&
|
this.activeFilter.selectedFolder &&
|
||||||
this.activeFilter.selectedFolderId != "none" &&
|
this.activeFilter.selectedFolderId != "none" &&
|
||||||
cipherPassesFilter
|
cipherPassesFilter
|
||||||
) {
|
) {
|
||||||
@@ -352,7 +352,7 @@ export class IndividualVaultComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
async editCipherId(id: string) {
|
async editCipherId(id: string) {
|
||||||
const cipher = await this.cipherService.get(id);
|
const cipher = await this.cipherService.get(id);
|
||||||
if (cipher.reprompt != 0) {
|
if (cipher != null && cipher.reprompt != 0) {
|
||||||
if (!(await this.passwordRepromptService.showPasswordPrompt())) {
|
if (!(await this.passwordRepromptService.showPasswordPrompt())) {
|
||||||
this.go({ cipherId: null });
|
this.go({ cipherId: null });
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ export class OrganizationVaultComponent implements OnInit, OnDestroy {
|
|||||||
this.ciphersComponent.organization = this.organization;
|
this.ciphersComponent.organization = this.organization;
|
||||||
|
|
||||||
this.route.queryParams.pipe(first()).subscribe(async (qParams) => {
|
this.route.queryParams.pipe(first()).subscribe(async (qParams) => {
|
||||||
// this.ciphersComponent.searchText = this.vaultFilterComponent.search = qParams.search;
|
this.ciphersComponent.searchText = this.vaultFilterComponent.searchText = qParams.search;
|
||||||
if (!this.organization.canViewAllCollections) {
|
if (!this.organization.canViewAllCollections) {
|
||||||
await this.syncService.fullSync(false);
|
await this.syncService.fullSync(false);
|
||||||
this.broadcasterService.subscribe(BroadcasterSubscriptionId, (message: any) => {
|
this.broadcasterService.subscribe(BroadcasterSubscriptionId, (message: any) => {
|
||||||
@@ -123,7 +123,11 @@ export class OrganizationVaultComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
this.route.queryParams.subscribe(async (params) => {
|
this.route.queryParams.subscribe(async (params) => {
|
||||||
if (params.cipherId) {
|
if (params.cipherId) {
|
||||||
if ((await this.cipherService.get(params.cipherId)) != null) {
|
if (
|
||||||
|
// Handle users with implicit collection access since they use the admin endpoint
|
||||||
|
this.organization.canEditAnyCollection ||
|
||||||
|
(await this.cipherService.get(params.cipherId)) != null
|
||||||
|
) {
|
||||||
this.editCipherId(params.cipherId);
|
this.editCipherId(params.cipherId);
|
||||||
} else {
|
} else {
|
||||||
this.platformUtilsService.showToast(
|
this.platformUtilsService.showToast(
|
||||||
@@ -168,7 +172,7 @@ export class OrganizationVaultComponent implements OnInit, OnDestroy {
|
|||||||
cipherPassesFilter = cipher.type === this.activeFilter.cipherType;
|
cipherPassesFilter = cipher.type === this.activeFilter.cipherType;
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
this.activeFilter.selectedFolderId != null &&
|
this.activeFilter.selectedFolder != null &&
|
||||||
this.activeFilter.selectedFolderId != "none" &&
|
this.activeFilter.selectedFolderId != "none" &&
|
||||||
cipherPassesFilter
|
cipherPassesFilter
|
||||||
) {
|
) {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Injectable } from "@angular/core";
|
import { Injectable } from "@angular/core";
|
||||||
import { ActivatedRouteSnapshot, CanActivate, Router } from "@angular/router";
|
import { ActivatedRouteSnapshot, CanActivate, Router, RouterStateSnapshot } from "@angular/router";
|
||||||
|
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
||||||
@@ -17,7 +17,7 @@ export class PermissionsGuard implements CanActivate {
|
|||||||
private syncService: SyncService
|
private syncService: SyncService
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async canActivate(route: ActivatedRouteSnapshot) {
|
async canActivate(route: ActivatedRouteSnapshot, state: RouterStateSnapshot) {
|
||||||
// TODO: We need to fix this issue once and for all.
|
// TODO: We need to fix this issue once and for all.
|
||||||
if ((await this.syncService.getLastSync()) == null) {
|
if ((await this.syncService.getLastSync()) == null) {
|
||||||
await this.syncService.fullSync(false);
|
await this.syncService.fullSync(false);
|
||||||
@@ -39,6 +39,16 @@ export class PermissionsGuard implements CanActivate {
|
|||||||
|
|
||||||
const permissions = route.data == null ? [] : (route.data.permissions as Permissions[]);
|
const permissions = route.data == null ? [] : (route.data.permissions as Permissions[]);
|
||||||
if (permissions != null && !org.hasAnyPermission(permissions)) {
|
if (permissions != null && !org.hasAnyPermission(permissions)) {
|
||||||
|
// Handle linkable ciphers for organizations the user only has view access to
|
||||||
|
// https://bitwarden.atlassian.net/browse/EC-203
|
||||||
|
if (state.root.queryParamMap.has("cipherId")) {
|
||||||
|
return this.router.createUrlTree(["/vault"], {
|
||||||
|
queryParams: {
|
||||||
|
cipherId: state.root.queryParamMap.get("cipherId"),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
this.platformUtilsService.showToast("error", null, this.i18nService.t("accessDenied"));
|
this.platformUtilsService.showToast("error", null, this.i18nService.t("accessDenied"));
|
||||||
return this.router.createUrlTree(["/"]);
|
return this.router.createUrlTree(["/"]);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
<app-navbar></app-navbar>
|
||||||
<div class="org-nav" *ngIf="organization">
|
<div class="org-nav" *ngIf="organization">
|
||||||
<div class="container d-flex">
|
<div class="container d-flex">
|
||||||
<div class="d-flex flex-column">
|
<div class="d-flex flex-column">
|
||||||
@@ -35,3 +36,4 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<router-outlet></router-outlet>
|
<router-outlet></router-outlet>
|
||||||
|
<app-footer></app-footer>
|
||||||
|
|||||||
@@ -20,8 +20,9 @@ import {
|
|||||||
import { ListResponse } from "jslib-common/models/response/listResponse";
|
import { ListResponse } from "jslib-common/models/response/listResponse";
|
||||||
import { CollectionView } from "jslib-common/models/view/collectionView";
|
import { CollectionView } from "jslib-common/models/view/collectionView";
|
||||||
|
|
||||||
|
import { EntityUsersComponent } from "../../modules/organizations/manage/entity-users.component";
|
||||||
|
|
||||||
import { CollectionAddEditComponent } from "./collection-add-edit.component";
|
import { CollectionAddEditComponent } from "./collection-add-edit.component";
|
||||||
import { EntityUsersComponent } from "./entity-users.component";
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: "app-org-manage-collections",
|
selector: "app-org-manage-collections",
|
||||||
|
|||||||
@@ -12,7 +12,8 @@ import { SearchService } from "jslib-common/abstractions/search.service";
|
|||||||
import { Utils } from "jslib-common/misc/utils";
|
import { Utils } from "jslib-common/misc/utils";
|
||||||
import { GroupResponse } from "jslib-common/models/response/groupResponse";
|
import { GroupResponse } from "jslib-common/models/response/groupResponse";
|
||||||
|
|
||||||
import { EntityUsersComponent } from "./entity-users.component";
|
import { EntityUsersComponent } from "../../modules/organizations/manage/entity-users.component";
|
||||||
|
|
||||||
import { GroupAddEditComponent } from "./group-add-edit.component";
|
import { GroupAddEditComponent } from "./group-add-edit.component";
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
|
|||||||
@@ -0,0 +1,117 @@
|
|||||||
|
<div class="modal fade" role="dialog" aria-modal="true" aria-labelledby="billingSyncApiKeyTitle">
|
||||||
|
<div class="modal-dialog modal-dialog-scrollable" role="document">
|
||||||
|
<form
|
||||||
|
class="modal-content"
|
||||||
|
#form
|
||||||
|
(ngSubmit)="submit()"
|
||||||
|
[appApiAction]="formPromise"
|
||||||
|
ngNativeValidate
|
||||||
|
>
|
||||||
|
<div class="modal-header">
|
||||||
|
<h2 class="modal-title" id="billingSyncApiKeyTitle">
|
||||||
|
{{ (hasBillingToken ? "viewBillingSyncToken" : "generateBillingSyncToken") | i18n }}
|
||||||
|
</h2>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="close"
|
||||||
|
data-dismiss="modal"
|
||||||
|
appA11yTitle="{{ 'close' | i18n }}"
|
||||||
|
>
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<app-user-verification
|
||||||
|
[(ngModel)]="masterPassword"
|
||||||
|
ngDefaultControl
|
||||||
|
name="secret"
|
||||||
|
*ngIf="!clientSecret"
|
||||||
|
>
|
||||||
|
</app-user-verification>
|
||||||
|
<ng-container *ngIf="clientSecret && showRotateScreen">
|
||||||
|
<p>{{ "rotateBillingSyncTokenTitle" | i18n }}</p>
|
||||||
|
<app-callout type="warning">
|
||||||
|
{{ "rotateBillingSyncTokenWarning" | i18n }}
|
||||||
|
</app-callout>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<div *ngIf="clientSecret && !showRotateScreen">
|
||||||
|
<p>{{ "copyPasteBillingSync" | i18n }}</p>
|
||||||
|
<label for="clientSecret">Billing Sync Key</label>
|
||||||
|
<div class="input-group">
|
||||||
|
<input
|
||||||
|
id="clientSecret"
|
||||||
|
class="form-control text-monospace"
|
||||||
|
type="text"
|
||||||
|
[(ngModel)]="clientSecret"
|
||||||
|
name="clientSecret"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<div class="input-group-append">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
(click)="copy()"
|
||||||
|
[appA11yTitle]="'copy' | i18n"
|
||||||
|
>
|
||||||
|
<i class="bwi bwi-lg bwi-clone" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="small text-muted mt-2" *ngIf="showLastSyncText">
|
||||||
|
<b class="font-weight-semibold">{{ "lastSync" | i18n }}:</b>
|
||||||
|
{{ lastSyncDate | date: "medium" }}
|
||||||
|
</div>
|
||||||
|
<div class="small text-danger mt-2" *ngIf="showAwaitingSyncText">
|
||||||
|
<i class="bwi bwi-error"></i>
|
||||||
|
{{
|
||||||
|
(daysBetween === 1 ? "awaitingSyncSingular" : "awaitingSyncPlural")
|
||||||
|
| i18n: daysBetween
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
class="btn btn-primary btn-submit"
|
||||||
|
[disabled]="form.loading"
|
||||||
|
*ngIf="!clientSecret || showRotateScreen"
|
||||||
|
>
|
||||||
|
<i
|
||||||
|
class="bwi bwi-spinner bwi-spin"
|
||||||
|
title="{{ 'loading' | i18n }}"
|
||||||
|
*ngIf="form.loading"
|
||||||
|
></i>
|
||||||
|
<span>
|
||||||
|
{{ submitButtonText }}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
data-dismiss="modal"
|
||||||
|
*ngIf="!showRotateScreen"
|
||||||
|
>
|
||||||
|
{{ "close" | i18n }}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
*ngIf="showRotateScreen"
|
||||||
|
(click)="cancelRotate()"
|
||||||
|
>
|
||||||
|
{{ "cancel" | i18n }}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
*ngIf="clientSecret && !showRotateScreen"
|
||||||
|
(click)="rotateToken()"
|
||||||
|
>
|
||||||
|
{{ "rotateToken" | i18n }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
108
src/app/organizations/settings/billing-sync-api-key.component.ts
Normal file
108
src/app/organizations/settings/billing-sync-api-key.component.ts
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
import { Component } from "@angular/core";
|
||||||
|
|
||||||
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
|
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
||||||
|
import { UserVerificationService } from "jslib-common/abstractions/userVerification.service";
|
||||||
|
import { OrganizationApiKeyType } from "jslib-common/enums/organizationApiKeyType";
|
||||||
|
import { OrganizationApiKeyRequest } from "jslib-common/models/request/organizationApiKeyRequest";
|
||||||
|
import { ApiKeyResponse } from "jslib-common/models/response/apiKeyResponse";
|
||||||
|
import { Verification } from "jslib-common/types/verification";
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: "app-billing-sync-api-key",
|
||||||
|
templateUrl: "billing-sync-api-key.component.html",
|
||||||
|
})
|
||||||
|
export class BillingSyncApiKeyComponent {
|
||||||
|
organizationId: string;
|
||||||
|
hasBillingToken: boolean;
|
||||||
|
|
||||||
|
showRotateScreen: boolean;
|
||||||
|
masterPassword: Verification;
|
||||||
|
formPromise: Promise<ApiKeyResponse>;
|
||||||
|
clientSecret?: string;
|
||||||
|
keyRevisionDate?: Date;
|
||||||
|
lastSyncDate?: Date = null;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private userVerificationService: UserVerificationService,
|
||||||
|
private apiService: ApiService,
|
||||||
|
private platformUtilsService: PlatformUtilsService,
|
||||||
|
private i18nService: I18nService
|
||||||
|
) {}
|
||||||
|
|
||||||
|
copy() {
|
||||||
|
this.platformUtilsService.copyToClipboard(this.clientSecret);
|
||||||
|
}
|
||||||
|
|
||||||
|
async submit() {
|
||||||
|
if (this.showRotateScreen) {
|
||||||
|
this.formPromise = this.userVerificationService
|
||||||
|
.buildRequest(this.masterPassword, OrganizationApiKeyRequest)
|
||||||
|
.then((request) => {
|
||||||
|
request.type = OrganizationApiKeyType.BillingSync;
|
||||||
|
return this.apiService.postOrganizationRotateApiKey(this.organizationId, request);
|
||||||
|
});
|
||||||
|
const response = await this.formPromise;
|
||||||
|
await this.load(response);
|
||||||
|
this.showRotateScreen = false;
|
||||||
|
this.platformUtilsService.showToast(
|
||||||
|
"success",
|
||||||
|
null,
|
||||||
|
this.i18nService.t("billingSyncApiKeyRotated")
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
this.formPromise = this.userVerificationService
|
||||||
|
.buildRequest(this.masterPassword, OrganizationApiKeyRequest)
|
||||||
|
.then((request) => {
|
||||||
|
request.type = OrganizationApiKeyType.BillingSync;
|
||||||
|
return this.apiService.postOrganizationApiKey(this.organizationId, request);
|
||||||
|
});
|
||||||
|
const response = await this.formPromise;
|
||||||
|
await this.load(response);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async load(response: ApiKeyResponse) {
|
||||||
|
this.clientSecret = response.apiKey;
|
||||||
|
this.keyRevisionDate = response.revisionDate;
|
||||||
|
this.hasBillingToken = true;
|
||||||
|
const syncStatus = await this.apiService.getSponsorshipSyncStatus(this.organizationId);
|
||||||
|
this.lastSyncDate = syncStatus.lastSyncDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
cancelRotate() {
|
||||||
|
this.showRotateScreen = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
rotateToken() {
|
||||||
|
this.showRotateScreen = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private dayDiff(date1: Date, date2: Date): number {
|
||||||
|
const diffTime = Math.abs(date2.getTime() - date1.getTime());
|
||||||
|
return Math.round(diffTime / (1000 * 60 * 60 * 24));
|
||||||
|
}
|
||||||
|
|
||||||
|
get submitButtonText(): string {
|
||||||
|
if (this.showRotateScreen) {
|
||||||
|
return this.i18nService.t("rotateToken");
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.i18nService.t(this.hasBillingToken ? "continue" : "generateToken");
|
||||||
|
}
|
||||||
|
|
||||||
|
get showLastSyncText(): boolean {
|
||||||
|
// If the keyRevisionDate is later than the lastSyncDate we need to show
|
||||||
|
// a warning that they need to put the billing sync key in their self hosted install
|
||||||
|
return this.lastSyncDate && this.lastSyncDate > this.keyRevisionDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
get showAwaitingSyncText(): boolean {
|
||||||
|
return this.lastSyncDate && this.lastSyncDate <= this.keyRevisionDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
get daysBetween(): number {
|
||||||
|
return this.dayDiff(this.keyRevisionDate, new Date());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -188,10 +188,10 @@
|
|||||||
></app-adjust-storage>
|
></app-adjust-storage>
|
||||||
</div>
|
</div>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
<!--Switch to i18n-->
|
||||||
<h2 class="spaced-header">{{ "additionalOptions" | i18n }}</h2>
|
<h2 class="spaced-header">{{ "selfHostingTitle" | i18n }}</h2>
|
||||||
<p class="mb-4">
|
<p class="mb-4">
|
||||||
{{ "additionalOptionsDesc" | i18n }}
|
{{ "selfHostingEnterpriseOrganizationSectionCopy" | i18n }}
|
||||||
</p>
|
</p>
|
||||||
<div class="d-flex">
|
<div class="d-flex">
|
||||||
<button
|
<button
|
||||||
@@ -203,6 +203,27 @@
|
|||||||
>
|
>
|
||||||
{{ "downloadLicense" | i18n }}
|
{{ "downloadLicense" | i18n }}
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary ml-1"
|
||||||
|
(click)="manageBillingSync()"
|
||||||
|
*ngIf="canManageBillingSync"
|
||||||
|
>
|
||||||
|
{{ (hasBillingSyncToken ? "manageBillingSync" : "setUpBillingSync") | i18n }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="mt-3" *ngIf="showDownloadLicense">
|
||||||
|
<app-download-license
|
||||||
|
[organizationId]="organizationId"
|
||||||
|
(onDownloaded)="closeDownloadLicense()"
|
||||||
|
(onCanceled)="closeDownloadLicense()"
|
||||||
|
></app-download-license>
|
||||||
|
</div>
|
||||||
|
<h2 class="spaced-header">{{ "additionalOptions" | i18n }}</h2>
|
||||||
|
<p class="mb-4">
|
||||||
|
{{ "additionalOptionsDesc" | i18n }}
|
||||||
|
</p>
|
||||||
|
<div class="d-flex">
|
||||||
<button
|
<button
|
||||||
#cancelBtn
|
#cancelBtn
|
||||||
type="button"
|
type="button"
|
||||||
@@ -216,13 +237,6 @@
|
|||||||
<span>{{ "cancelSubscription" | i18n }}</span>
|
<span>{{ "cancelSubscription" | i18n }}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-3" *ngIf="showDownloadLicense">
|
|
||||||
<app-download-license
|
|
||||||
[organizationId]="organizationId"
|
|
||||||
(onDownloaded)="closeDownloadLicense()"
|
|
||||||
(onCanceled)="closeDownloadLicense()"
|
|
||||||
></app-download-license>
|
|
||||||
</div>
|
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-container *ngIf="selfHosted">
|
<ng-container *ngIf="selfHosted">
|
||||||
<dl>
|
<dl>
|
||||||
@@ -269,5 +283,31 @@
|
|||||||
></app-update-license>
|
></app-update-license>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div *ngIf="showBillingSyncKey">
|
||||||
|
<h2 class="mt-5">
|
||||||
|
{{ "billingSync" | i18n }}
|
||||||
|
</h2>
|
||||||
|
<p>
|
||||||
|
{{ "billingSyncDesc" | i18n }}
|
||||||
|
</p>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-outline-secondary"
|
||||||
|
(click)="manageBillingSyncSelfHosted()"
|
||||||
|
>
|
||||||
|
{{ "manageBillingSync" | i18n }}
|
||||||
|
</button>
|
||||||
|
<small class="form-text text-muted" *ngIf="billingSyncSetUp">
|
||||||
|
{{ "lastSync" | i18n }}:
|
||||||
|
<span *ngIf="userOrg.familySponsorshipLastSyncDate != null">
|
||||||
|
{{ userOrg.familySponsorshipLastSyncDate | date: "medium" }}
|
||||||
|
</span>
|
||||||
|
<span *ngIf="userOrg.familySponsorshipLastSyncDate == null">
|
||||||
|
{{ "never" | i18n | lowercase }}
|
||||||
|
</span>
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
<ng-template #setupBillingSyncTemplate></ng-template>
|
||||||
|
<ng-template #rotateBillingSyncKeyTemplate></ng-template>
|
||||||
|
|||||||
@@ -1,21 +1,34 @@
|
|||||||
import { Component, OnInit } from "@angular/core";
|
import { Component, OnInit, ViewChild, ViewContainerRef } from "@angular/core";
|
||||||
import { ActivatedRoute } from "@angular/router";
|
import { ActivatedRoute } from "@angular/router";
|
||||||
|
|
||||||
|
import { ModalRef } from "jslib-angular/components/modal/modal.ref";
|
||||||
|
import { ModalService } from "jslib-angular/services/modal.service";
|
||||||
import { ApiService } from "jslib-common/abstractions/api.service";
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { LogService } from "jslib-common/abstractions/log.service";
|
import { LogService } from "jslib-common/abstractions/log.service";
|
||||||
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
||||||
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
||||||
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
||||||
|
import { OrganizationApiKeyType } from "jslib-common/enums/organizationApiKeyType";
|
||||||
|
import { OrganizationConnectionType } from "jslib-common/enums/organizationConnectionType";
|
||||||
import { PlanType } from "jslib-common/enums/planType";
|
import { PlanType } from "jslib-common/enums/planType";
|
||||||
|
import { BillingSyncConfigApi } from "jslib-common/models/api/billingSyncConfigApi";
|
||||||
import { Organization } from "jslib-common/models/domain/organization";
|
import { Organization } from "jslib-common/models/domain/organization";
|
||||||
|
import { OrganizationConnectionResponse } from "jslib-common/models/response/organizationConnectionResponse";
|
||||||
import { OrganizationSubscriptionResponse } from "jslib-common/models/response/organizationSubscriptionResponse";
|
import { OrganizationSubscriptionResponse } from "jslib-common/models/response/organizationSubscriptionResponse";
|
||||||
|
|
||||||
|
import { BillingSyncKeyComponent } from "src/app/settings/billing-sync-key.component";
|
||||||
|
|
||||||
|
import { BillingSyncApiKeyComponent } from "./billing-sync-api-key.component";
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: "app-org-subscription",
|
selector: "app-org-subscription",
|
||||||
templateUrl: "organization-subscription.component.html",
|
templateUrl: "organization-subscription.component.html",
|
||||||
})
|
})
|
||||||
export class OrganizationSubscriptionComponent implements OnInit {
|
export class OrganizationSubscriptionComponent implements OnInit {
|
||||||
|
@ViewChild("setupBillingSyncTemplate", { read: ViewContainerRef, static: true })
|
||||||
|
setupBillingSyncModalRef: ViewContainerRef;
|
||||||
|
|
||||||
loading = false;
|
loading = false;
|
||||||
firstLoaded = false;
|
firstLoaded = false;
|
||||||
organizationId: string;
|
organizationId: string;
|
||||||
@@ -25,17 +38,24 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
adjustStorageAdd = true;
|
adjustStorageAdd = true;
|
||||||
showAdjustStorage = false;
|
showAdjustStorage = false;
|
||||||
showUpdateLicense = false;
|
showUpdateLicense = false;
|
||||||
|
showBillingSyncKey = false;
|
||||||
showDownloadLicense = false;
|
showDownloadLicense = false;
|
||||||
showChangePlan = false;
|
showChangePlan = false;
|
||||||
sub: OrganizationSubscriptionResponse;
|
sub: OrganizationSubscriptionResponse;
|
||||||
selfHosted = false;
|
selfHosted = false;
|
||||||
|
hasBillingSyncToken: boolean;
|
||||||
|
|
||||||
userOrg: Organization;
|
userOrg: Organization;
|
||||||
|
existingBillingSyncConnection: OrganizationConnectionResponse<BillingSyncConfigApi>;
|
||||||
|
|
||||||
removeSponsorshipPromise: Promise<any>;
|
removeSponsorshipPromise: Promise<any>;
|
||||||
cancelPromise: Promise<any>;
|
cancelPromise: Promise<any>;
|
||||||
reinstatePromise: Promise<any>;
|
reinstatePromise: Promise<any>;
|
||||||
|
|
||||||
|
@ViewChild("rotateBillingSyncKeyTemplate", { read: ViewContainerRef, static: true })
|
||||||
|
billingSyncKeyViewContainerRef: ViewContainerRef;
|
||||||
|
billingSyncKeyRef: [ModalRef, BillingSyncKeyComponent];
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private apiService: ApiService,
|
private apiService: ApiService,
|
||||||
private platformUtilsService: PlatformUtilsService,
|
private platformUtilsService: PlatformUtilsService,
|
||||||
@@ -43,7 +63,8 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
private messagingService: MessagingService,
|
private messagingService: MessagingService,
|
||||||
private route: ActivatedRoute,
|
private route: ActivatedRoute,
|
||||||
private organizationService: OrganizationService,
|
private organizationService: OrganizationService,
|
||||||
private logService: LogService
|
private logService: LogService,
|
||||||
|
private modalService: ModalService
|
||||||
) {
|
) {
|
||||||
this.selfHosted = platformUtilsService.isSelfHost();
|
this.selfHosted = platformUtilsService.isSelfHost();
|
||||||
}
|
}
|
||||||
@@ -63,10 +84,27 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
|
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
this.userOrg = await this.organizationService.get(this.organizationId);
|
this.userOrg = await this.organizationService.get(this.organizationId);
|
||||||
|
|
||||||
if (this.userOrg.canManageBilling) {
|
if (this.userOrg.canManageBilling) {
|
||||||
this.sub = await this.apiService.getOrganizationSubscription(this.organizationId);
|
this.sub = await this.apiService.getOrganizationSubscription(this.organizationId);
|
||||||
}
|
}
|
||||||
|
const apiKeyResponse = await this.apiService.getOrganizationApiKeyInformation(
|
||||||
|
this.organizationId
|
||||||
|
);
|
||||||
|
this.hasBillingSyncToken = apiKeyResponse.data.some(
|
||||||
|
(i) => i.keyType === OrganizationApiKeyType.BillingSync
|
||||||
|
);
|
||||||
|
|
||||||
|
if (this.selfHosted) {
|
||||||
|
this.showBillingSyncKey = await this.apiService.getCloudCommunicationsEnabled();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.showBillingSyncKey) {
|
||||||
|
this.existingBillingSyncConnection = await this.apiService.getOrganizationConnection(
|
||||||
|
this.organizationId,
|
||||||
|
OrganizationConnectionType.CloudBillingSync,
|
||||||
|
BillingSyncConfigApi
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
}
|
}
|
||||||
@@ -138,6 +176,20 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
this.showDownloadLicense = !this.showDownloadLicense;
|
this.showDownloadLicense = !this.showDownloadLicense;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async manageBillingSync() {
|
||||||
|
const [ref] = await this.modalService.openViewRef(
|
||||||
|
BillingSyncApiKeyComponent,
|
||||||
|
this.setupBillingSyncModalRef,
|
||||||
|
(comp) => {
|
||||||
|
comp.organizationId = this.organizationId;
|
||||||
|
comp.hasBillingToken = this.hasBillingSyncToken;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
ref.onClosed.subscribe(async () => {
|
||||||
|
await this.load();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
closeDownloadLicense() {
|
closeDownloadLicense() {
|
||||||
this.showDownloadLicense = false;
|
this.showDownloadLicense = false;
|
||||||
}
|
}
|
||||||
@@ -200,6 +252,24 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async manageBillingSyncSelfHosted() {
|
||||||
|
this.billingSyncKeyRef = await this.modalService.openViewRef(
|
||||||
|
BillingSyncKeyComponent,
|
||||||
|
this.billingSyncKeyViewContainerRef,
|
||||||
|
(comp) => {
|
||||||
|
comp.entityId = this.organizationId;
|
||||||
|
comp.existingConnectionId = this.existingBillingSyncConnection?.id;
|
||||||
|
comp.billingSyncKey = this.existingBillingSyncConnection?.config?.billingSyncKey;
|
||||||
|
comp.setParentConnection = (
|
||||||
|
connection: OrganizationConnectionResponse<BillingSyncConfigApi>
|
||||||
|
) => {
|
||||||
|
this.existingBillingSyncConnection = connection;
|
||||||
|
this.billingSyncKeyRef[0].close();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
get isExpired() {
|
get isExpired() {
|
||||||
return (
|
return (
|
||||||
this.sub != null && this.sub.expiration != null && new Date(this.sub.expiration) < new Date()
|
this.sub != null && this.sub.expiration != null && new Date(this.sub.expiration) < new Date()
|
||||||
@@ -266,6 +336,16 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get canManageBillingSync() {
|
||||||
|
return (
|
||||||
|
!this.selfHosted &&
|
||||||
|
(this.sub.planType === PlanType.EnterpriseAnnually ||
|
||||||
|
this.sub.planType === PlanType.EnterpriseMonthly ||
|
||||||
|
this.sub.planType === PlanType.EnterpriseAnnually2019 ||
|
||||||
|
this.sub.planType === PlanType.EnterpriseMonthly2019)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
get subscriptionDesc() {
|
get subscriptionDesc() {
|
||||||
if (this.sub.planType === PlanType.Free) {
|
if (this.sub.planType === PlanType.Free) {
|
||||||
return this.i18nService.t("subscriptionFreePlan", this.sub.seats.toString());
|
return this.i18nService.t("subscriptionFreePlan", this.sub.seats.toString());
|
||||||
@@ -293,4 +373,8 @@ export class OrganizationSubscriptionComponent implements OnInit {
|
|||||||
get showChangePlanButton() {
|
get showChangePlanButton() {
|
||||||
return this.subscription == null && this.sub.planType === PlanType.Free && !this.showChangePlan;
|
return this.subscription == null && this.sub.planType === PlanType.Free && !this.showChangePlan;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get billingSyncSetUp() {
|
||||||
|
return this.existingBillingSyncConnection?.id != null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ import { CipherView } from "jslib-common/models/view/cipherView";
|
|||||||
import { ExposedPasswordsReportComponent as BaseExposedPasswordsReportComponent } from "../../reports/exposed-passwords-report.component";
|
import { ExposedPasswordsReportComponent as BaseExposedPasswordsReportComponent } from "../../reports/exposed-passwords-report.component";
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: "app-exposed-passwords-report",
|
selector: "app-org-exposed-passwords-report",
|
||||||
templateUrl: "../../reports/exposed-passwords-report.component.html",
|
templateUrl: "../../reports/exposed-passwords-report.component.html",
|
||||||
})
|
})
|
||||||
export class ExposedPasswordsReportComponent extends BaseExposedPasswordsReportComponent {
|
export class ExposedPasswordsReportComponent extends BaseExposedPasswordsReportComponent {
|
||||||
@@ -41,12 +41,10 @@ export class ExposedPasswordsReportComponent extends BaseExposedPasswordsReportC
|
|||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
const dynamicSuper = Object.getPrototypeOf(this.constructor.prototype);
|
|
||||||
this.route.parent.parent.params.subscribe(async (params) => {
|
this.route.parent.parent.params.subscribe(async (params) => {
|
||||||
this.organization = await this.organizationService.get(params.organizationId);
|
this.organization = await this.organizationService.get(params.organizationId);
|
||||||
this.manageableCiphers = await this.cipherService.getAll();
|
this.manageableCiphers = await this.cipherService.getAll();
|
||||||
// TODO: We should do something about this, calling super in an async function is bad
|
await this.checkAccess();
|
||||||
dynamicSuper.ngOnInit();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -58,7 +58,8 @@ export class CiphersComponent extends BaseCiphersComponent {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async load(filter: (cipher: CipherView) => boolean = null) {
|
async load(filter: (cipher: CipherView) => boolean = null, deleted = false) {
|
||||||
|
this.deleted = deleted || false;
|
||||||
if (this.organization.canEditAnyCollection) {
|
if (this.organization.canEditAnyCollection) {
|
||||||
this.accessEvents = this.organization.useEvents;
|
this.accessEvents = this.organization.useEvents;
|
||||||
this.allCiphers = await this.cipherService.getAllFromApiForOrganization(this.organization.id);
|
this.allCiphers = await this.cipherService.getAllFromApiForOrganization(this.organization.id);
|
||||||
|
|||||||
@@ -155,6 +155,11 @@ const routes: Routes = [
|
|||||||
.IndividualVaultModule,
|
.IndividualVaultModule,
|
||||||
},
|
},
|
||||||
{ path: "sends", component: SendComponent, data: { title: "Send" } },
|
{ path: "sends", component: SendComponent, data: { title: "Send" } },
|
||||||
|
{
|
||||||
|
path: "create-organization",
|
||||||
|
component: CreateOrganizationComponent,
|
||||||
|
data: { titleId: "newOrganization" },
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: "settings",
|
path: "settings",
|
||||||
component: SettingsComponent,
|
component: SettingsComponent,
|
||||||
@@ -181,11 +186,6 @@ const routes: Routes = [
|
|||||||
loadChildren: async () =>
|
loadChildren: async () =>
|
||||||
(await import("./settings/subscription-routing.module")).SubscriptionRoutingModule,
|
(await import("./settings/subscription-routing.module")).SubscriptionRoutingModule,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
path: "create-organization",
|
|
||||||
component: CreateOrganizationComponent,
|
|
||||||
data: { titleId: "newOrganization" },
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
path: "emergency-access",
|
path: "emergency-access",
|
||||||
children: [
|
children: [
|
||||||
@@ -229,15 +229,15 @@ const routes: Routes = [
|
|||||||
(await import("./reports/reports-routing.module")).ReportsRoutingModule,
|
(await import("./reports/reports-routing.module")).ReportsRoutingModule,
|
||||||
},
|
},
|
||||||
{ path: "setup/families-for-enterprise", component: FamiliesForEnterpriseSetupComponent },
|
{ path: "setup/families-for-enterprise", component: FamiliesForEnterpriseSetupComponent },
|
||||||
{
|
|
||||||
path: "organizations",
|
|
||||||
loadChildren: () =>
|
|
||||||
import("./organizations/organization-routing.module").then(
|
|
||||||
(m) => m.OrganizationsRoutingModule
|
|
||||||
),
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: "organizations",
|
||||||
|
loadChildren: () =>
|
||||||
|
import("./organizations/organization-routing.module").then(
|
||||||
|
(m) => m.OrganizationsRoutingModule
|
||||||
|
),
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { NgModule } from "@angular/core";
|
import { NgModule } from "@angular/core";
|
||||||
|
|
||||||
import { LooseComponentsModule } from "./modules/loose-components.module";
|
import { LooseComponentsModule } from "./modules/loose-components.module";
|
||||||
|
import { OrganizationManageModule } from "./modules/organizations/manage/organization-manage.module";
|
||||||
import { PipesModule } from "./modules/pipes/pipes.module";
|
import { PipesModule } from "./modules/pipes/pipes.module";
|
||||||
import { SharedModule } from "./modules/shared.module";
|
import { SharedModule } from "./modules/shared.module";
|
||||||
import { VaultFilterModule } from "./modules/vault-filter/vault-filter.module";
|
import { VaultFilterModule } from "./modules/vault-filter/vault-filter.module";
|
||||||
@@ -13,6 +14,7 @@ import { OrganizationBadgeModule } from "./modules/vault/modules/organization-ba
|
|||||||
VaultFilterModule,
|
VaultFilterModule,
|
||||||
OrganizationBadgeModule,
|
OrganizationBadgeModule,
|
||||||
PipesModule,
|
PipesModule,
|
||||||
|
OrganizationManageModule,
|
||||||
],
|
],
|
||||||
exports: [LooseComponentsModule, VaultFilterModule, OrganizationBadgeModule, PipesModule],
|
exports: [LooseComponentsModule, VaultFilterModule, OrganizationBadgeModule, PipesModule],
|
||||||
bootstrap: [],
|
bootstrap: [],
|
||||||
|
|||||||
69
src/app/settings/billing-sync-key.component.html
Normal file
69
src/app/settings/billing-sync-key.component.html
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
<div class="modal fade" role="dialog" aria-modal="true" aria-labelledby="billingSyncTitle">
|
||||||
|
<div class="modal-dialog modal-dialog-scrollable" role="document">
|
||||||
|
<form
|
||||||
|
class="modal-content"
|
||||||
|
#form
|
||||||
|
(ngSubmit)="submit()"
|
||||||
|
[appApiAction]="formPromise"
|
||||||
|
ngNativeValidate
|
||||||
|
>
|
||||||
|
<div class="modal-header">
|
||||||
|
<h2 class="modal-title" id="billingSyncTitle">{{ "manageBillingSync" | i18n }}</h2>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="close"
|
||||||
|
data-dismiss="modal"
|
||||||
|
appA11yTitle="{{ 'close' | i18n }}"
|
||||||
|
>
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<p>{{ "billingSyncKeyDesc" | i18n }}</p>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="billingSyncKey"
|
||||||
|
>{{ "billingSyncKey" | i18n }} <small>(</small><small>{{ "required" | i18n }}</small
|
||||||
|
><small>)</small></label
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
id="billingSyncKey"
|
||||||
|
type="input"
|
||||||
|
name="billingSyncKey"
|
||||||
|
class="form-control"
|
||||||
|
[(ngModel)]="billingSyncKey"
|
||||||
|
required
|
||||||
|
appAutofocus
|
||||||
|
appInputVerbatim
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<button type="submit" class="btn btn-primary btn-submit" [disabled]="form.loading">
|
||||||
|
<i class="bwi bwi-spinner bwi-spin" title="{{ 'loading' | i18n }}" aria-hidden="true"></i>
|
||||||
|
<span>{{ "save" | i18n }}</span>
|
||||||
|
</button>
|
||||||
|
<button type="button" class="btn btn-outline-secondary" data-dismiss="modal">
|
||||||
|
{{ "cancel" | i18n }}
|
||||||
|
</button>
|
||||||
|
<div class="ml-auto">
|
||||||
|
<button
|
||||||
|
#deleteBtn
|
||||||
|
type="button"
|
||||||
|
(click)="deleteConnection()"
|
||||||
|
class="btn btn-outline-danger"
|
||||||
|
appA11yTitle="{{ 'delete' | i18n }}"
|
||||||
|
[disabled]="form.loading"
|
||||||
|
>
|
||||||
|
<i class="bwi bwi-trash bwi-lg bwi-fw" [hidden]="form.loading" aria-hidden="true"></i>
|
||||||
|
<i
|
||||||
|
class="bwi bwi-spinner bwi-spin bwi-lg bwi-fw"
|
||||||
|
[hidden]="!form.loading"
|
||||||
|
title="{{ 'loading' | i18n }}"
|
||||||
|
aria-hidden="true"
|
||||||
|
></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
61
src/app/settings/billing-sync-key.component.ts
Normal file
61
src/app/settings/billing-sync-key.component.ts
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
import { Component } from "@angular/core";
|
||||||
|
|
||||||
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
|
import { LogService } from "jslib-common/abstractions/log.service";
|
||||||
|
import { OrganizationConnectionType } from "jslib-common/enums/organizationConnectionType";
|
||||||
|
import { Utils } from "jslib-common/misc/utils";
|
||||||
|
import { BillingSyncConfigApi } from "jslib-common/models/api/billingSyncConfigApi";
|
||||||
|
import { BillingSyncConfigRequest } from "jslib-common/models/request/billingSyncConfigRequest";
|
||||||
|
import { OrganizationConnectionRequest } from "jslib-common/models/request/organizationConnectionRequest";
|
||||||
|
import { OrganizationConnectionResponse } from "jslib-common/models/response/organizationConnectionResponse";
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: "app-billing-sync-key",
|
||||||
|
templateUrl: "billing-sync-key.component.html",
|
||||||
|
})
|
||||||
|
export class BillingSyncKeyComponent {
|
||||||
|
entityId: string;
|
||||||
|
existingConnectionId: string;
|
||||||
|
billingSyncKey: string;
|
||||||
|
setParentConnection: (connection: OrganizationConnectionResponse<BillingSyncConfigApi>) => void;
|
||||||
|
|
||||||
|
formPromise: Promise<OrganizationConnectionResponse<BillingSyncConfigApi>> | Promise<void>;
|
||||||
|
|
||||||
|
constructor(private apiService: ApiService, private logService: LogService) {}
|
||||||
|
|
||||||
|
async submit() {
|
||||||
|
try {
|
||||||
|
const request = new OrganizationConnectionRequest(
|
||||||
|
this.entityId,
|
||||||
|
OrganizationConnectionType.CloudBillingSync,
|
||||||
|
true,
|
||||||
|
new BillingSyncConfigRequest(this.billingSyncKey)
|
||||||
|
);
|
||||||
|
if (this.existingConnectionId == null) {
|
||||||
|
this.formPromise = this.apiService.createOrganizationConnection(
|
||||||
|
request,
|
||||||
|
BillingSyncConfigApi
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
this.formPromise = this.apiService.updateOrganizationConnection(
|
||||||
|
request,
|
||||||
|
BillingSyncConfigApi,
|
||||||
|
this.existingConnectionId
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const response = (await this
|
||||||
|
.formPromise) as OrganizationConnectionResponse<BillingSyncConfigApi>;
|
||||||
|
this.existingConnectionId = response?.id;
|
||||||
|
this.billingSyncKey = response?.config?.billingSyncKey;
|
||||||
|
this.setParentConnection(response);
|
||||||
|
} catch (e) {
|
||||||
|
this.logService.error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async deleteConnection() {
|
||||||
|
this.formPromise = this.apiService.deleteOrganizationConnection(this.existingConnectionId);
|
||||||
|
await this.formPromise;
|
||||||
|
this.setParentConnection(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -68,6 +68,7 @@ export class ChangePasswordComponent extends BaseChangePasswordComponent {
|
|||||||
if (await this.keyConnectorService.getUsesKeyConnector()) {
|
if (await this.keyConnectorService.getUsesKeyConnector()) {
|
||||||
this.router.navigate(["/settings/security/two-factor"]);
|
this.router.navigate(["/settings/security/two-factor"]);
|
||||||
}
|
}
|
||||||
|
await super.ngOnInit();
|
||||||
}
|
}
|
||||||
|
|
||||||
async rotateEncKeyClicked() {
|
async rotateEncKeyClicked() {
|
||||||
|
|||||||
@@ -1,5 +1,11 @@
|
|||||||
<div class="page-header">
|
<div class="container page-content">
|
||||||
<h1>{{ "newOrganization" | i18n }}</h1>
|
<div class="row">
|
||||||
|
<div class="col-12">
|
||||||
|
<div class="page-header">
|
||||||
|
<h1>{{ "newOrganization" | i18n }}</h1>
|
||||||
|
</div>
|
||||||
|
<p>{{ "newOrganizationDesc" | i18n }}</p>
|
||||||
|
<app-organization-plans></app-organization-plans>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p>{{ "newOrganizationDesc" | i18n }}</p>
|
|
||||||
<app-organization-plans></app-organization-plans>
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { ApiService } from "jslib-common/abstractions/api.service";
|
|||||||
import { CryptoService } from "jslib-common/abstractions/crypto.service";
|
import { CryptoService } from "jslib-common/abstractions/crypto.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { LogService } from "jslib-common/abstractions/log.service";
|
import { LogService } from "jslib-common/abstractions/log.service";
|
||||||
|
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
||||||
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
||||||
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
||||||
import { PolicyService } from "jslib-common/abstractions/policy.service";
|
import { PolicyService } from "jslib-common/abstractions/policy.service";
|
||||||
@@ -68,7 +69,8 @@ export class OrganizationPlansComponent implements OnInit {
|
|||||||
private syncService: SyncService,
|
private syncService: SyncService,
|
||||||
private policyService: PolicyService,
|
private policyService: PolicyService,
|
||||||
private organizationService: OrganizationService,
|
private organizationService: OrganizationService,
|
||||||
private logService: LogService
|
private logService: LogService,
|
||||||
|
private messagingService: MessagingService
|
||||||
) {
|
) {
|
||||||
this.selfHosted = platformUtilsService.isSelfHost();
|
this.selfHosted = platformUtilsService.isSelfHost();
|
||||||
}
|
}
|
||||||
@@ -298,6 +300,7 @@ export class OrganizationPlansComponent implements OnInit {
|
|||||||
this.formPromise = doSubmit();
|
this.formPromise = doSubmit();
|
||||||
const organizationId = await this.formPromise;
|
const organizationId = await this.formPromise;
|
||||||
this.onSuccess.emit({ organizationId: organizationId });
|
this.onSuccess.emit({ organizationId: organizationId });
|
||||||
|
this.messagingService.send("organizationCreated", organizationId);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,7 +54,11 @@ export class SettingsComponent implements OnInit, OnDestroy {
|
|||||||
this.premium = await this.tokenService.getPremium();
|
this.premium = await this.tokenService.getPremium();
|
||||||
this.hasFamilySponsorshipAvailable = await this.organizationService.canManageSponsorships();
|
this.hasFamilySponsorshipAvailable = await this.organizationService.canManageSponsorships();
|
||||||
const hasPremiumFromOrg = await this.stateService.getCanAccessPremium();
|
const hasPremiumFromOrg = await this.stateService.getCanAccessPremium();
|
||||||
const billing = await this.apiService.getUserBillingHistory();
|
let billing = null;
|
||||||
this.hideSubscription = !this.premium && hasPremiumFromOrg && billing.hasNoHistory;
|
if (!this.selfHosted) {
|
||||||
|
billing = await this.apiService.getUserBillingHistory();
|
||||||
|
}
|
||||||
|
this.hideSubscription =
|
||||||
|
!this.premium && hasPremiumFromOrg && (this.selfHosted || billing?.hasNoHistory);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,33 +20,55 @@
|
|||||||
#form
|
#form
|
||||||
(ngSubmit)="submit()"
|
(ngSubmit)="submit()"
|
||||||
[appApiAction]="formPromise"
|
[appApiAction]="formPromise"
|
||||||
|
[formGroup]="sponsorshipForm"
|
||||||
ngNativeValidate
|
ngNativeValidate
|
||||||
*ngIf="anyOrgsAvailable"
|
*ngIf="anyOrgsAvailable"
|
||||||
>
|
>
|
||||||
<div *ngIf="moreThanOneOrgAvailable" class="form-group col-7">
|
<div class="form-group col-7">
|
||||||
<label for="availableSponsorshipOrg">{{ "familiesSponsoringOrgSelect" | i18n }}</label>
|
<label for="availableSponsorshipOrg">{{ "familiesSponsoringOrgSelect" | i18n }}</label>
|
||||||
<select
|
<select
|
||||||
id="availableSponsorshipOrg"
|
id="availableSponsorshipOrg"
|
||||||
name="Available Sponsorship Organization"
|
name="Available Sponsorship Organization"
|
||||||
[(ngModel)]="selectedSponsorshipOrgId"
|
formControlName="selectedSponsorshipOrgId"
|
||||||
class="form-control"
|
class="form-control"
|
||||||
required
|
required
|
||||||
>
|
>
|
||||||
<option value="">-- {{ "select" | i18n }} --</option>
|
<option disabled="true" value="">-- {{ "select" | i18n }} --</option>
|
||||||
<option *ngFor="let o of availableSponsorshipOrgs" [ngValue]="o.id">{{ o.name }}</option>
|
<option *ngFor="let o of availableSponsorshipOrgs" [ngValue]="o.id">{{ o.name }}</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group col-7">
|
<div class="form-group col-7">
|
||||||
<label for="accountEmail">{{ "sponsoredFamiliesEmail" | i18n }}:</label>
|
<label for="sponsorshipEmail">{{ "sponsoredFamiliesEmail" | i18n }}:</label>
|
||||||
<input
|
<input
|
||||||
id="accountEmail"
|
id="sponsorshipEmail"
|
||||||
class="form-control"
|
class="form-control"
|
||||||
inputmode="email"
|
inputmode="email"
|
||||||
[(ngModel)]="sponsorshipEmail"
|
formControlName="sponsorshipEmail"
|
||||||
name="sponsorshipEmail"
|
name="sponsorshipEmail"
|
||||||
required
|
required
|
||||||
|
[attr.aria-invalid]="sponsorshipEmailControl.invalid"
|
||||||
/>
|
/>
|
||||||
<button class="btn btn-primary btn-submit mt-4" type="submit" [disabled]="form.loading">
|
<small
|
||||||
|
aria-errormessage="sponsorshipEmail"
|
||||||
|
*ngIf="sponsorshipEmailControl.errors?.notAllowedValue"
|
||||||
|
class="error-inline"
|
||||||
|
role="alert"
|
||||||
|
>
|
||||||
|
<i class="bwi bwi-error" aria-hidden="true"></i>
|
||||||
|
{{ "cannotSponsorSelf" | i18n }}
|
||||||
|
</small>
|
||||||
|
<small
|
||||||
|
aria-errormessage="sponsorshipEmail"
|
||||||
|
*ngIf="sponsorshipEmailControl.errors?.email"
|
||||||
|
class="error-inline"
|
||||||
|
role="alert"
|
||||||
|
>
|
||||||
|
<i class="bwi bwi-error" aria-hidden="true"></i>
|
||||||
|
{{ "invalidEmail" | i18n }}
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
|
<div class="form-group col-7">
|
||||||
|
<button class="btn btn-primary btn-submit mt-2" type="submit" [disabled]="form.loading">
|
||||||
<i class="bwi bwi-spinner bwi-spin" title="{{ 'loading' | i18n }}" aria-hidden="true"></i>
|
<i class="bwi bwi-spinner bwi-spin" title="{{ 'loading' | i18n }}" aria-hidden="true"></i>
|
||||||
<span>{{ "redeem" | i18n }}</span>
|
<span>{{ "redeem" | i18n }}</span>
|
||||||
</button>
|
</button>
|
||||||
@@ -59,12 +81,18 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<th>{{ "recipient" | i18n }}</th>
|
<th>{{ "recipient" | i18n }}</th>
|
||||||
<th>{{ "sponsoringOrg" | i18n }}</th>
|
<th>{{ "sponsoringOrg" | i18n }}</th>
|
||||||
|
<th>{{ "status" | i18n }}</th>
|
||||||
<th></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<ng-container *ngFor="let o of activeSponsorshipOrgs">
|
<ng-container *ngFor="let o of activeSponsorshipOrgs">
|
||||||
<tr sponsoring-org-row [sponsoringOrg]="o" (sponsorshipRemoved)="load(true)"></tr>
|
<tr
|
||||||
|
sponsoring-org-row
|
||||||
|
[sponsoringOrg]="o"
|
||||||
|
[isSelfHosted]="isSelfHosted"
|
||||||
|
(sponsorshipRemoved)="load(true)"
|
||||||
|
></tr>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -1,9 +1,12 @@
|
|||||||
import { Component, OnInit } from "@angular/core";
|
import { Component, OnInit } from "@angular/core";
|
||||||
|
import { FormBuilder, FormGroup, Validators } from "@angular/forms";
|
||||||
|
|
||||||
|
import { notAllowedValueAsync } from "jslib-angular/validators/notAllowedValueAsync.validator";
|
||||||
import { ApiService } from "jslib-common/abstractions/api.service";
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
import { OrganizationService } from "jslib-common/abstractions/organization.service";
|
||||||
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
||||||
|
import { StateService } from "jslib-common/abstractions/state.service";
|
||||||
import { SyncService } from "jslib-common/abstractions/sync.service";
|
import { SyncService } from "jslib-common/abstractions/sync.service";
|
||||||
import { PlanSponsorshipType } from "jslib-common/enums/planSponsorshipType";
|
import { PlanSponsorshipType } from "jslib-common/enums/planSponsorshipType";
|
||||||
import { Organization } from "jslib-common/models/domain/organization";
|
import { Organization } from "jslib-common/models/domain/organization";
|
||||||
@@ -17,30 +20,54 @@ export class SponsoredFamiliesComponent implements OnInit {
|
|||||||
|
|
||||||
availableSponsorshipOrgs: Organization[] = [];
|
availableSponsorshipOrgs: Organization[] = [];
|
||||||
activeSponsorshipOrgs: Organization[] = [];
|
activeSponsorshipOrgs: Organization[] = [];
|
||||||
selectedSponsorshipOrgId = "";
|
|
||||||
sponsorshipEmail = "";
|
|
||||||
|
|
||||||
// Conditional display properties
|
// Conditional display properties
|
||||||
formPromise: Promise<any>;
|
formPromise: Promise<any>;
|
||||||
|
|
||||||
|
sponsorshipForm: FormGroup;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private apiService: ApiService,
|
private apiService: ApiService,
|
||||||
private i18nService: I18nService,
|
private i18nService: I18nService,
|
||||||
private platformUtilsService: PlatformUtilsService,
|
private platformUtilsService: PlatformUtilsService,
|
||||||
private syncService: SyncService,
|
private syncService: SyncService,
|
||||||
private organizationService: OrganizationService
|
private organizationService: OrganizationService,
|
||||||
) {}
|
private formBuilder: FormBuilder,
|
||||||
|
private stateService: StateService
|
||||||
|
) {
|
||||||
|
this.sponsorshipForm = this.formBuilder.group({
|
||||||
|
selectedSponsorshipOrgId: [
|
||||||
|
"",
|
||||||
|
{
|
||||||
|
validators: [Validators.required],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
sponsorshipEmail: [
|
||||||
|
"",
|
||||||
|
{
|
||||||
|
validators: [Validators.email],
|
||||||
|
asyncValidators: [
|
||||||
|
notAllowedValueAsync(async () => await this.stateService.getEmail(), true),
|
||||||
|
],
|
||||||
|
updateOn: "blur",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async ngOnInit() {
|
async ngOnInit() {
|
||||||
await this.load();
|
await this.load();
|
||||||
}
|
}
|
||||||
|
|
||||||
async submit() {
|
async submit() {
|
||||||
this.formPromise = this.apiService.postCreateSponsorship(this.selectedSponsorshipOrgId, {
|
this.formPromise = this.apiService.postCreateSponsorship(
|
||||||
sponsoredEmail: this.sponsorshipEmail,
|
this.sponsorshipForm.value.selectedSponsorshipOrgId,
|
||||||
planSponsorshipType: PlanSponsorshipType.FamiliesForEnterprise,
|
{
|
||||||
friendlyName: this.sponsorshipEmail,
|
sponsoredEmail: this.sponsorshipForm.value.sponsorshipEmail,
|
||||||
});
|
planSponsorshipType: PlanSponsorshipType.FamiliesForEnterprise,
|
||||||
|
friendlyName: this.sponsorshipForm.value.sponsorshipEmail,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
await this.formPromise;
|
await this.formPromise;
|
||||||
this.platformUtilsService.showToast("success", null, this.i18nService.t("sponsorshipCreated"));
|
this.platformUtilsService.showToast("success", null, this.i18nService.t("sponsorshipCreated"));
|
||||||
@@ -67,14 +94,19 @@ export class SponsoredFamiliesComponent implements OnInit {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (this.availableSponsorshipOrgs.length === 1) {
|
if (this.availableSponsorshipOrgs.length === 1) {
|
||||||
this.selectedSponsorshipOrgId = this.availableSponsorshipOrgs[0].id;
|
this.sponsorshipForm.patchValue({
|
||||||
|
selectedSponsorshipOrgId: this.availableSponsorshipOrgs[0].id,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get sponsorshipEmailControl() {
|
||||||
|
return this.sponsorshipForm.controls["sponsorshipEmail"];
|
||||||
|
}
|
||||||
|
|
||||||
private async resetForm() {
|
private async resetForm() {
|
||||||
this.sponsorshipEmail = "";
|
this.sponsorshipForm.reset();
|
||||||
this.selectedSponsorshipOrgId = "";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get anyActiveSponsorships(): boolean {
|
get anyActiveSponsorships(): boolean {
|
||||||
@@ -85,7 +117,7 @@ export class SponsoredFamiliesComponent implements OnInit {
|
|||||||
return this.availableSponsorshipOrgs.length > 0;
|
return this.availableSponsorshipOrgs.length > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
get moreThanOneOrgAvailable(): boolean {
|
get isSelfHosted(): boolean {
|
||||||
return this.availableSponsorshipOrgs.length > 1;
|
return this.platformUtilsService.isSelfHost();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,9 +2,13 @@
|
|||||||
{{ sponsoringOrg.familySponsorshipFriendlyName }}
|
{{ sponsoringOrg.familySponsorshipFriendlyName }}
|
||||||
</td>
|
</td>
|
||||||
<td>{{ sponsoringOrg.name }}</td>
|
<td>{{ sponsoringOrg.name }}</td>
|
||||||
|
<td>
|
||||||
|
<span [ngClass]="statusClass">{{ statusMessage }}</span>
|
||||||
|
</td>
|
||||||
<td class="table-action-right">
|
<td class="table-action-right">
|
||||||
<div class="dropdown" appListDropdown>
|
<div class="dropdown" appListDropdown>
|
||||||
<button
|
<button
|
||||||
|
*ngIf="!sponsoringOrg.familySponsorshipToDelete"
|
||||||
class="btn btn-outline-secondary dropdown-toggle"
|
class="btn btn-outline-secondary dropdown-toggle"
|
||||||
type="button"
|
type="button"
|
||||||
id="dropdownMenuButton"
|
id="dropdownMenuButton"
|
||||||
@@ -18,6 +22,7 @@
|
|||||||
<div class="dropdown-menu dropdown-menu-right" aria-labelledby="dropdownMenuButton">
|
<div class="dropdown-menu dropdown-menu-right" aria-labelledby="dropdownMenuButton">
|
||||||
<button
|
<button
|
||||||
#resendEmailBtn
|
#resendEmailBtn
|
||||||
|
*ngIf="!isSelfHosted && !sponsoringOrg.familySponsorshipValidUntil"
|
||||||
[appApiAction]="resendEmailPromise"
|
[appApiAction]="resendEmailPromise"
|
||||||
class="dropdown-item btn-submit"
|
class="dropdown-item btn-submit"
|
||||||
[disabled]="resendEmailBtn.loading"
|
[disabled]="resendEmailBtn.loading"
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Component, EventEmitter, Input, Output } from "@angular/core";
|
import { formatDate } from "@angular/common";
|
||||||
|
import { Component, EventEmitter, Input, Output, OnInit } from "@angular/core";
|
||||||
|
|
||||||
import { ApiService } from "jslib-common/abstractions/api.service";
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
||||||
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
import { I18nService } from "jslib-common/abstractions/i18n.service";
|
||||||
@@ -10,11 +11,15 @@ import { Organization } from "jslib-common/models/domain/organization";
|
|||||||
selector: "[sponsoring-org-row]",
|
selector: "[sponsoring-org-row]",
|
||||||
templateUrl: "sponsoring-org-row.component.html",
|
templateUrl: "sponsoring-org-row.component.html",
|
||||||
})
|
})
|
||||||
export class SponsoringOrgRowComponent {
|
export class SponsoringOrgRowComponent implements OnInit {
|
||||||
@Input() sponsoringOrg: Organization = null;
|
@Input() sponsoringOrg: Organization = null;
|
||||||
|
@Input() isSelfHosted = false;
|
||||||
|
|
||||||
@Output() sponsorshipRemoved = new EventEmitter();
|
@Output() sponsorshipRemoved = new EventEmitter();
|
||||||
|
|
||||||
|
statusMessage = "loading";
|
||||||
|
statusClass: "text-success" | "text-danger" = "text-success";
|
||||||
|
|
||||||
revokeSponsorshipPromise: Promise<any>;
|
revokeSponsorshipPromise: Promise<any>;
|
||||||
resendEmailPromise: Promise<any>;
|
resendEmailPromise: Promise<any>;
|
||||||
|
|
||||||
@@ -25,6 +30,15 @@ export class SponsoringOrgRowComponent {
|
|||||||
private platformUtilsService: PlatformUtilsService
|
private platformUtilsService: PlatformUtilsService
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
ngOnInit(): void {
|
||||||
|
this.setStatus(
|
||||||
|
this.isSelfHosted,
|
||||||
|
this.sponsoringOrg.familySponsorshipToDelete,
|
||||||
|
this.sponsoringOrg.familySponsorshipValidUntil,
|
||||||
|
this.sponsoringOrg.familySponsorshipLastSyncDate
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
async revokeSponsorship() {
|
async revokeSponsorship() {
|
||||||
try {
|
try {
|
||||||
this.revokeSponsorshipPromise = this.doRevokeSponsorship();
|
this.revokeSponsorshipPromise = this.doRevokeSponsorship();
|
||||||
@@ -43,6 +57,10 @@ export class SponsoringOrgRowComponent {
|
|||||||
this.resendEmailPromise = null;
|
this.resendEmailPromise = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get isSentAwaitingSync() {
|
||||||
|
return this.isSelfHosted && !this.sponsoringOrg.familySponsorshipLastSyncDate;
|
||||||
|
}
|
||||||
|
|
||||||
private async doRevokeSponsorship() {
|
private async doRevokeSponsorship() {
|
||||||
const isConfirmed = await this.platformUtilsService.showDialog(
|
const isConfirmed = await this.platformUtilsService.showDialog(
|
||||||
this.i18nService.t("revokeSponsorshipConfirmation"),
|
this.i18nService.t("revokeSponsorshipConfirmation"),
|
||||||
@@ -60,4 +78,53 @@ export class SponsoringOrgRowComponent {
|
|||||||
this.platformUtilsService.showToast("success", null, this.i18nService.t("reclaimedFreePlan"));
|
this.platformUtilsService.showToast("success", null, this.i18nService.t("reclaimedFreePlan"));
|
||||||
this.sponsorshipRemoved.emit();
|
this.sponsorshipRemoved.emit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private setStatus(
|
||||||
|
selfHosted: boolean,
|
||||||
|
toDelete?: boolean,
|
||||||
|
validUntil?: Date,
|
||||||
|
lastSyncDate?: Date
|
||||||
|
) {
|
||||||
|
/*
|
||||||
|
* Possible Statuses:
|
||||||
|
* Requested (self-hosted only)
|
||||||
|
* Sent
|
||||||
|
* Active
|
||||||
|
* RequestRevoke
|
||||||
|
* RevokeWhenExpired
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (toDelete && validUntil) {
|
||||||
|
// They want to delete but there is a valid until date which means there is an active sponsorship
|
||||||
|
this.statusMessage = this.i18nService.t(
|
||||||
|
"revokeWhenExpired",
|
||||||
|
formatDate(validUntil, "MM/dd/yyyy", this.i18nService.locale)
|
||||||
|
);
|
||||||
|
this.statusClass = "text-danger";
|
||||||
|
} else if (toDelete) {
|
||||||
|
// They want to delete and we don't have a valid until date so we can
|
||||||
|
// this should only happen on a self-hosted install
|
||||||
|
this.statusMessage = this.i18nService.t("requestRemoved");
|
||||||
|
this.statusClass = "text-danger";
|
||||||
|
} else if (validUntil) {
|
||||||
|
// They don't want to delete and they have a valid until date
|
||||||
|
// that means they are actively sponsoring someone
|
||||||
|
this.statusMessage = this.i18nService.t("active");
|
||||||
|
this.statusClass = "text-success";
|
||||||
|
} else if (selfHosted && lastSyncDate) {
|
||||||
|
// We are on a self-hosted install and it has been synced but we have not gotten
|
||||||
|
// a valid until date so we can't know if they are actively sponsoring someone
|
||||||
|
this.statusMessage = this.i18nService.t("sent");
|
||||||
|
this.statusClass = "text-success";
|
||||||
|
} else if (!selfHosted) {
|
||||||
|
// We are in cloud and all other status checks have been false therefore we have
|
||||||
|
// sent the request but it hasn't been accepted yet
|
||||||
|
this.statusMessage = this.i18nService.t("sent");
|
||||||
|
this.statusClass = "text-success";
|
||||||
|
} else {
|
||||||
|
// We are on a self-hosted install and we have not synced yet
|
||||||
|
this.statusMessage = this.i18nService.t("requested");
|
||||||
|
this.statusClass = "text-success";
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,10 +73,14 @@ export class TaxInfoComponent {
|
|||||||
this.logService.error(e);
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const taxInfo = await this.apiService.getTaxInfo();
|
try {
|
||||||
if (taxInfo) {
|
const taxInfo = await this.apiService.getTaxInfo();
|
||||||
this.taxInfo.postalCode = taxInfo.postalCode;
|
if (taxInfo) {
|
||||||
this.taxInfo.country = taxInfo.country || "US";
|
this.taxInfo.postalCode = taxInfo.postalCode;
|
||||||
|
this.taxInfo.country = taxInfo.country || "US";
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
this.logService.error(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.pristine = Object.assign({}, this.taxInfo);
|
this.pristine = Object.assign({}, this.taxInfo);
|
||||||
@@ -86,9 +90,16 @@ export class TaxInfoComponent {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const taxRates = await this.apiService.getTaxRates();
|
try {
|
||||||
this.taxRates = taxRates.data;
|
const taxRates = await this.apiService.getTaxRates();
|
||||||
this.loading = false;
|
if (taxRates) {
|
||||||
|
this.taxRates = taxRates.data;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
this.logService.error(e);
|
||||||
|
} finally {
|
||||||
|
this.loading = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get taxRate() {
|
get taxRate() {
|
||||||
|
|||||||
@@ -295,16 +295,6 @@
|
|||||||
(blur)="saveUsernameOptions()"
|
(blur)="saveUsernameOptions()"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group col-4">
|
|
||||||
<label for="simplelogin-hostname">{{ "hostname" | i18n }}</label>
|
|
||||||
<input
|
|
||||||
id="simplelogin-hostname"
|
|
||||||
class="form-control"
|
|
||||||
type="text"
|
|
||||||
[(ngModel)]="usernameOptions.forwardedSimpleLoginHostname"
|
|
||||||
(blur)="saveUsernameOptions()"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="row" *ngIf="usernameOptions.forwardedService === 'anonaddy'">
|
<div class="row" *ngIf="usernameOptions.forwardedService === 'anonaddy'">
|
||||||
<div class="form-group col-4">
|
<div class="form-group col-4">
|
||||||
|
|||||||
@@ -50,7 +50,8 @@
|
|||||||
organizationName="{{ c.organizationId | orgNameFromId: organizations }}"
|
organizationName="{{ c.organizationId | orgNameFromId: organizations }}"
|
||||||
profileName="{{ profileName }}"
|
profileName="{{ profileName }}"
|
||||||
(onOrganizationClicked)="onOrganizationClicked(c.organizationId)"
|
(onOrganizationClicked)="onOrganizationClicked(c.organizationId)"
|
||||||
></app-org-badge>
|
>
|
||||||
|
</app-org-badge>
|
||||||
</td>
|
</td>
|
||||||
<td class="table-list-options">
|
<td class="table-list-options">
|
||||||
<button
|
<button
|
||||||
|
|||||||
@@ -77,7 +77,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<a
|
<a
|
||||||
href="#"
|
href="#"
|
||||||
routerLink="/settings/create-organization"
|
routerLink="/create-organization"
|
||||||
class="btn btn-primary"
|
class="btn btn-primary"
|
||||||
*ngIf="!organizations || !organizations.length"
|
*ngIf="!organizations || !organizations.length"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -916,7 +916,7 @@
|
|||||||
},
|
},
|
||||||
"specialCharacters": {
|
"specialCharacters": {
|
||||||
"message": "Special Characters (!@#$%^&*)"
|
"message": "Special Characters (!@#$%^&*)"
|
||||||
},
|
},
|
||||||
"numWords": {
|
"numWords": {
|
||||||
"message": "Number of Words"
|
"message": "Number of Words"
|
||||||
},
|
},
|
||||||
@@ -3034,10 +3034,10 @@
|
|||||||
"message": "Adjustments to your subscription will result in prorated changes to your billing totals. You cannot invite more than $COUNT$ users without increasing your subscription seats.",
|
"message": "Adjustments to your subscription will result in prorated changes to your billing totals. You cannot invite more than $COUNT$ users without increasing your subscription seats.",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
"count": {
|
"count": {
|
||||||
"content": "$1",
|
"content": "$1",
|
||||||
"example": "50"
|
"example": "50"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"seatsToAdd": {
|
"seatsToAdd": {
|
||||||
"message": "Seats To Add"
|
"message": "Seats To Add"
|
||||||
@@ -4395,10 +4395,10 @@
|
|||||||
"message": "Your Master Password does not meet the policy requirements of this organization. In order to join the organization, you must update your Master Password now. Proceeding will log you out of your current session, requiring you to log back in. Active sessions on other devices may continue to remain active for up to one hour."
|
"message": "Your Master Password does not meet the policy requirements of this organization. In order to join the organization, you must update your Master Password now. Proceeding will log you out of your current session, requiring you to log back in. Active sessions on other devices may continue to remain active for up to one hour."
|
||||||
},
|
},
|
||||||
"maximumVaultTimeout": {
|
"maximumVaultTimeout": {
|
||||||
"message": "Vault Timeout"
|
"message": "Vault Timeout"
|
||||||
},
|
},
|
||||||
"maximumVaultTimeoutDesc": {
|
"maximumVaultTimeoutDesc": {
|
||||||
"message": "Configure a maximum vault timeout for all users."
|
"message": "Configure a maximum vault timeout for all users."
|
||||||
},
|
},
|
||||||
"maximumVaultTimeoutLabel": {
|
"maximumVaultTimeoutLabel": {
|
||||||
"message": "Maximum Vault Timeout"
|
"message": "Maximum Vault Timeout"
|
||||||
@@ -4594,7 +4594,7 @@
|
|||||||
"message": "Enter your personal email to redeem Bitwarden Families"
|
"message": "Enter your personal email to redeem Bitwarden Families"
|
||||||
},
|
},
|
||||||
"sponsoredFamiliesLeaveCopy": {
|
"sponsoredFamiliesLeaveCopy": {
|
||||||
"message": "If you leave or are removed from the sponsoring organization, your Families plan will expire at the end of the billing period."
|
"message": "If you remove an offer or are removed from the sponsoring organization, your Families sponsorship will expire at the next renewal date."
|
||||||
},
|
},
|
||||||
"acceptBitwardenFamiliesHelp": {
|
"acceptBitwardenFamiliesHelp": {
|
||||||
"message": "Accept offer for an existing organization or create a new Families organization."
|
"message": "Accept offer for an existing organization or create a new Families organization."
|
||||||
@@ -4669,7 +4669,7 @@
|
|||||||
"message": "Email Sent"
|
"message": "Email Sent"
|
||||||
},
|
},
|
||||||
"revokeSponsorshipConfirmation": {
|
"revokeSponsorshipConfirmation": {
|
||||||
"message": "After removing this account, the Families organization owner will be responsible for this subscription and related invoices. Are you sure you want to continue?"
|
"message": "After removing this account, the Families plan sponsorship will expire at the end of the billing period. You will not be able to redeem a new sponsorship offer until the existing one expires. Are you sure you want to continue?"
|
||||||
},
|
},
|
||||||
"removeSponsorshipSuccess": {
|
"removeSponsorshipSuccess": {
|
||||||
"message": "Sponsorship Removed"
|
"message": "Sponsorship Removed"
|
||||||
@@ -4803,6 +4803,75 @@
|
|||||||
"freeWithSponsorship": {
|
"freeWithSponsorship": {
|
||||||
"message": "FREE with sponsorship"
|
"message": "FREE with sponsorship"
|
||||||
},
|
},
|
||||||
|
"viewBillingSyncToken": {
|
||||||
|
"message": "View Billing Sync Token"
|
||||||
|
},
|
||||||
|
"generateBillingSyncToken": {
|
||||||
|
"message": "Generate Billing Sync Token"
|
||||||
|
},
|
||||||
|
"copyPasteBillingSync": {
|
||||||
|
"message": "Copy and paste this token into the Billing Sync settings of your self-hosted organization."
|
||||||
|
},
|
||||||
|
"billingSyncCanAccess": {
|
||||||
|
"message": "Your Billing Sync token can access and edit this organization's subscription settings."
|
||||||
|
},
|
||||||
|
"manageBillingSync": {
|
||||||
|
"message": "Manage Billing Sync"
|
||||||
|
},
|
||||||
|
"setUpBillingSync": {
|
||||||
|
"message": "Set Up Billing Sync"
|
||||||
|
},
|
||||||
|
"generateToken": {
|
||||||
|
"message": "Generate Token"
|
||||||
|
},
|
||||||
|
"rotateToken": {
|
||||||
|
"message": "Rotate Token"
|
||||||
|
},
|
||||||
|
"rotateBillingSyncTokenWarning": {
|
||||||
|
"message": "If you proceed, you will need to re-setup billing sync on your self-hosted server."
|
||||||
|
},
|
||||||
|
"rotateBillingSyncTokenTitle": {
|
||||||
|
"message": "Rotating the Billing Sync Token will invalidate the previous token."
|
||||||
|
},
|
||||||
|
"selfHostingTitle": {
|
||||||
|
"message": "Self-Hosting"
|
||||||
|
},
|
||||||
|
"selfHostingEnterpriseOrganizationSectionCopy": {
|
||||||
|
"message": "To set-up your organization on your own server, you will need to upload your license file. To support Free Families plans and advanced billing capabilities for your self-hosted organization, you will need to set up billing sync."
|
||||||
|
},
|
||||||
|
"billingSyncApiKeyRotated": {
|
||||||
|
"message": "Token rotated."
|
||||||
|
},
|
||||||
|
"billingSync": {
|
||||||
|
"message": "Billing Sync"
|
||||||
|
},
|
||||||
|
"billingSyncDesc": {
|
||||||
|
"message": "Billing Sync provides Free Families plans for members and advanced billing capabilities by linking your self-hosted Bitwarden to the Bitwarden cloud server."
|
||||||
|
},
|
||||||
|
"billingSyncKeyDesc": {
|
||||||
|
"message": "A Billing Sync Token from your cloud organization's subscription settings is required to complete this form."
|
||||||
|
},
|
||||||
|
"billingSyncKey": {
|
||||||
|
"message": "Billing Sync Token"
|
||||||
|
},
|
||||||
|
"active": {
|
||||||
|
"message": "Active"
|
||||||
|
},
|
||||||
|
"inactive": {
|
||||||
|
"message": "Inactive"
|
||||||
|
},
|
||||||
|
"sentAwaitingSync": {
|
||||||
|
"message": "Sent (Awaiting Sync)"
|
||||||
|
},
|
||||||
|
"sent": {
|
||||||
|
"message": "Sent"
|
||||||
|
},
|
||||||
|
"requestRemoved": {
|
||||||
|
"message": "Removed (Awaiting Sync)"
|
||||||
|
},
|
||||||
|
"requested": {
|
||||||
|
"message": "Requested"
|
||||||
|
},
|
||||||
"formErrorSummaryPlural": {
|
"formErrorSummaryPlural": {
|
||||||
"message": "$COUNT$ fields above need your attention.",
|
"message": "$COUNT$ fields above need your attention.",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -4935,6 +5004,43 @@
|
|||||||
"service": {
|
"service": {
|
||||||
"message": "Service"
|
"message": "Service"
|
||||||
},
|
},
|
||||||
|
"unknownCipher": {
|
||||||
|
"message": "Unknown Item, you may need to request permission to access this item."
|
||||||
|
},
|
||||||
|
"cannotSponsorSelf": {
|
||||||
|
"message": "You cannot redeem for the active account. Enter a different email."
|
||||||
|
},
|
||||||
|
"revokeWhenExpired": {
|
||||||
|
"message": "Expires $DATE$",
|
||||||
|
"placeholders": {
|
||||||
|
"date": {
|
||||||
|
"content": "$1",
|
||||||
|
"example": "12/31/2020"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"awaitingSyncSingular": {
|
||||||
|
"message": "Token rotated $DAYS$ day ago. Update the billing sync token in your self-hosted organization settings.",
|
||||||
|
"placeholders": {
|
||||||
|
"days": {
|
||||||
|
"content": "$1",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"awaitingSyncPlural": {
|
||||||
|
"message": "Token rotated $DAYS$ days ago. Update the billing sync token in your self-hosted organization settings.",
|
||||||
|
"placeholders": {
|
||||||
|
"days": {
|
||||||
|
"content": "$1",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"lastSync": {
|
||||||
|
"message": "Last Sync",
|
||||||
|
"Description": "Used as a prefix to indicate the last time a sync occured. Example \"Last sync 1968-11-16 00:00:00\""
|
||||||
|
},
|
||||||
"billingManagedByProvider": {
|
"billingManagedByProvider": {
|
||||||
"message": "Managed by $PROVIDER$",
|
"message": "Managed by $PROVIDER$",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -4960,8 +5066,5 @@
|
|||||||
},
|
},
|
||||||
"apiAccessToken": {
|
"apiAccessToken": {
|
||||||
"message": "API Access Token"
|
"message": "API Access Token"
|
||||||
},
|
|
||||||
"unknownCipher": {
|
|
||||||
"message": "Unknown Item, you may need to login with another account to access this item."
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -68,7 +68,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.modal-footer {
|
.modal-footer {
|
||||||
border-radius: 0.3rem 0.3rem 0 0;
|
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
@include themify($themes) {
|
@include themify($themes) {
|
||||||
background-color: themed("footerBackgroundColor");
|
background-color: themed("footerBackgroundColor");
|
||||||
|
|||||||
@@ -28,8 +28,6 @@
|
|||||||
|
|
||||||
h3,
|
h3,
|
||||||
button.filter-button {
|
button.filter-button {
|
||||||
text-transform: uppercase;
|
|
||||||
text-transform: uppercase;
|
|
||||||
margin: 0;
|
margin: 0;
|
||||||
@include themify($themes) {
|
@include themify($themes) {
|
||||||
color: themed("textMuted");
|
color: themed("textMuted");
|
||||||
@@ -118,6 +116,7 @@
|
|||||||
}
|
}
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
max-width: 90%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.edit-button {
|
.edit-button {
|
||||||
|
|||||||
@@ -204,8 +204,60 @@ const devServer =
|
|||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
key: "Content-Security-Policy",
|
key: "Content-Security-Policy",
|
||||||
value:
|
value: `
|
||||||
"default-src 'self'; script-src 'self' 'sha256-ryoU+5+IUZTuUyTElqkrQGBJXr1brEv6r2CA62WUw8w=' https://js.stripe.com https://js.braintreegateway.com https://www.paypalobjects.com; style-src 'self' https://assets.braintreegateway.com https://*.paypal.com 'sha256-47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=' 'sha256-JVRXyYPueLWdwGwY9m/7u4QlZ1xeQdqUj2t8OVIzZE4='; img-src 'self' data: https://icons.bitwarden.net https://*.paypal.com https://www.paypalobjects.com https://q.stripe.com https://haveibeenpwned.com https://www.gravatar.com; child-src 'self' https://js.stripe.com https://assets.braintreegateway.com https://*.paypal.com https://*.duosecurity.com; frame-src 'self' https://js.stripe.com https://assets.braintreegateway.com https://*.paypal.com https://*.duosecurity.com; connect-src 'self' wss://notifications.bitwarden.com https://notifications.bitwarden.com https://cdn.bitwarden.net https://api.pwnedpasswords.com https://2fa.directory/api/v3/totp.json https://api.stripe.com https://www.paypal.com https://api.braintreegateway.com https://client-analytics.braintreegateway.com https://*.braintree-api.com https://*.blob.core.windows.net https://app.simplelogin.io/api/alias/random/new https://app.anonaddy.com/api/v1/aliases; object-src 'self' blob:;",
|
default-src 'self';
|
||||||
|
script-src
|
||||||
|
'self'
|
||||||
|
'sha256-ryoU+5+IUZTuUyTElqkrQGBJXr1brEv6r2CA62WUw8w='
|
||||||
|
https://js.stripe.com
|
||||||
|
https://js.braintreegateway.com
|
||||||
|
https://www.paypalobjects.com;
|
||||||
|
style-src
|
||||||
|
'self'
|
||||||
|
https://assets.braintreegateway.com
|
||||||
|
https://*.paypal.com
|
||||||
|
'sha256-47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU='
|
||||||
|
'sha256-JVRXyYPueLWdwGwY9m/7u4QlZ1xeQdqUj2t8OVIzZE4=';
|
||||||
|
'sha256-0xHKHIT3+e2Gknxsm/cpErSprhL+o254L/y5bljg74U='
|
||||||
|
img-src
|
||||||
|
'self'
|
||||||
|
data:
|
||||||
|
https://icons.bitwarden.net
|
||||||
|
https://*.paypal.com
|
||||||
|
https://www.paypalobjects.com
|
||||||
|
https://q.stripe.com
|
||||||
|
https://haveibeenpwned.com
|
||||||
|
https://www.gravatar.com;
|
||||||
|
child-src
|
||||||
|
'self'
|
||||||
|
https://js.stripe.com
|
||||||
|
https://assets.braintreegateway.com
|
||||||
|
https://*.paypal.com
|
||||||
|
https://*.duosecurity.com;
|
||||||
|
frame-src
|
||||||
|
'self'
|
||||||
|
https://js.stripe.com
|
||||||
|
https://assets.braintreegateway.com
|
||||||
|
https://*.paypal.com
|
||||||
|
https://*.duosecurity.com;
|
||||||
|
connect-src
|
||||||
|
'self'
|
||||||
|
wss://notifications.bitwarden.com
|
||||||
|
https://notifications.bitwarden.com
|
||||||
|
https://cdn.bitwarden.net
|
||||||
|
https://api.pwnedpasswords.com
|
||||||
|
https://2fa.directory/api/v3/totp.json
|
||||||
|
https://api.stripe.com
|
||||||
|
https://www.paypal.com
|
||||||
|
https://api.braintreegateway.com
|
||||||
|
https://client-analytics.braintreegateway.com
|
||||||
|
https://*.braintree-api.com
|
||||||
|
https://*.blob.core.windows.net
|
||||||
|
https://app.simplelogin.io/api/alias/random/new
|
||||||
|
https://app.anonaddy.com/api/v1/aliases;
|
||||||
|
object-src
|
||||||
|
'self'
|
||||||
|
blob:;`,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user