Compare commits
2 Commits
1cb33a407d
...
1eaea2618a
Author | SHA1 | Date | |
---|---|---|---|
1eaea2618a | |||
|
a137bbbdca |
@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## Unreleased
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Improved the usability of the user account registration
|
||||||
|
|
||||||
## 2.145.1 - 2025-03-10
|
## 2.145.1 - 2025-03-10
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
@ -7,7 +7,6 @@ import { hasPermission, permissions } from '@ghostfolio/common/permissions';
|
|||||||
import { Component, OnDestroy, OnInit } from '@angular/core';
|
import { Component, OnDestroy, OnInit } from '@angular/core';
|
||||||
import { MatDialog } from '@angular/material/dialog';
|
import { MatDialog } from '@angular/material/dialog';
|
||||||
import { Router } from '@angular/router';
|
import { Router } from '@angular/router';
|
||||||
import { Role } from '@prisma/client';
|
|
||||||
import { DeviceDetectorService } from 'ngx-device-detector';
|
import { DeviceDetectorService } from 'ngx-device-detector';
|
||||||
import { Subject } from 'rxjs';
|
import { Subject } from 'rxjs';
|
||||||
import { takeUntil } from 'rxjs/operators';
|
import { takeUntil } from 'rxjs/operators';
|
||||||
@ -59,15 +58,6 @@ export class RegisterPageComponent implements OnDestroy, OnInit {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async createAccount() {
|
|
||||||
this.dataService
|
|
||||||
.postUser()
|
|
||||||
.pipe(takeUntil(this.unsubscribeSubject))
|
|
||||||
.subscribe(({ accessToken, authToken, role }) => {
|
|
||||||
this.openShowAccessTokenDialog(accessToken, authToken, role);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public async onLoginWithInternetIdentity() {
|
public async onLoginWithInternetIdentity() {
|
||||||
try {
|
try {
|
||||||
const { authToken } = await this.internetIdentityService.login();
|
const { authToken } = await this.internetIdentityService.login();
|
||||||
@ -76,17 +66,8 @@ export class RegisterPageComponent implements OnDestroy, OnInit {
|
|||||||
} catch {}
|
} catch {}
|
||||||
}
|
}
|
||||||
|
|
||||||
public openShowAccessTokenDialog(
|
public openShowAccessTokenDialog() {
|
||||||
accessToken: string,
|
|
||||||
authToken: string,
|
|
||||||
role: Role
|
|
||||||
) {
|
|
||||||
const dialogRef = this.dialog.open(ShowAccessTokenDialog, {
|
const dialogRef = this.dialog.open(ShowAccessTokenDialog, {
|
||||||
data: {
|
|
||||||
accessToken,
|
|
||||||
authToken,
|
|
||||||
role
|
|
||||||
},
|
|
||||||
disableClose: true,
|
disableClose: true,
|
||||||
width: '30rem'
|
width: '30rem'
|
||||||
});
|
});
|
||||||
@ -94,8 +75,8 @@ export class RegisterPageComponent implements OnDestroy, OnInit {
|
|||||||
dialogRef
|
dialogRef
|
||||||
.afterClosed()
|
.afterClosed()
|
||||||
.pipe(takeUntil(this.unsubscribeSubject))
|
.pipe(takeUntil(this.unsubscribeSubject))
|
||||||
.subscribe((data) => {
|
.subscribe((authToken) => {
|
||||||
if (data?.authToken) {
|
if (authToken) {
|
||||||
this.tokenStorageService.saveToken(authToken, true);
|
this.tokenStorageService.saveToken(authToken, true);
|
||||||
|
|
||||||
this.router.navigate(['/']);
|
this.router.navigate(['/']);
|
||||||
|
@ -22,7 +22,7 @@
|
|||||||
class="d-inline-block"
|
class="d-inline-block"
|
||||||
color="primary"
|
color="primary"
|
||||||
mat-flat-button
|
mat-flat-button
|
||||||
(click)="createAccount()"
|
(click)="openShowAccessTokenDialog()"
|
||||||
>
|
>
|
||||||
<ng-container i18n>Create Account</ng-container>
|
<ng-container i18n>Create Account</ng-container>
|
||||||
</button>
|
</button>
|
||||||
|
@ -1,19 +1,58 @@
|
|||||||
import { ChangeDetectionStrategy, Component, Inject } from '@angular/core';
|
import { DataService } from '@ghostfolio/client/services/data.service';
|
||||||
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
||||||
|
import {
|
||||||
|
ChangeDetectionStrategy,
|
||||||
|
ChangeDetectorRef,
|
||||||
|
Component,
|
||||||
|
ViewChild
|
||||||
|
} from '@angular/core';
|
||||||
|
import { MatStepper } from '@angular/material/stepper';
|
||||||
|
import { Subject } from 'rxjs';
|
||||||
|
import { takeUntil } from 'rxjs/operators';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'gf-show-access-token-dialog',
|
|
||||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||||
|
selector: 'gf-show-access-token-dialog',
|
||||||
|
standalone: false,
|
||||||
styleUrls: ['./show-access-token-dialog.scss'],
|
styleUrls: ['./show-access-token-dialog.scss'],
|
||||||
templateUrl: 'show-access-token-dialog.html',
|
templateUrl: 'show-access-token-dialog.html'
|
||||||
standalone: false
|
|
||||||
})
|
})
|
||||||
export class ShowAccessTokenDialog {
|
export class ShowAccessTokenDialog {
|
||||||
public isAgreeButtonDisabled = true;
|
@ViewChild(MatStepper) stepper!: MatStepper;
|
||||||
|
|
||||||
public constructor(@Inject(MAT_DIALOG_DATA) public data: any) {}
|
public accessToken: string;
|
||||||
|
public authToken: string;
|
||||||
|
public isCreateAccountButtonDisabled = true;
|
||||||
|
public isDisclaimerChecked = false;
|
||||||
|
public role: string;
|
||||||
|
|
||||||
public enableAgreeButton() {
|
private unsubscribeSubject = new Subject<void>();
|
||||||
this.isAgreeButtonDisabled = false;
|
|
||||||
|
public constructor(
|
||||||
|
private changeDetectorRef: ChangeDetectorRef,
|
||||||
|
private dataService: DataService
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public createAccount() {
|
||||||
|
this.dataService
|
||||||
|
.postUser()
|
||||||
|
.pipe(takeUntil(this.unsubscribeSubject))
|
||||||
|
.subscribe(({ accessToken, authToken, role }) => {
|
||||||
|
this.accessToken = accessToken;
|
||||||
|
this.authToken = authToken;
|
||||||
|
this.role = role;
|
||||||
|
|
||||||
|
this.stepper.next();
|
||||||
|
|
||||||
|
this.changeDetectorRef.markForCheck();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public enableCreateAccountButton() {
|
||||||
|
this.isCreateAccountButtonDisabled = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public onChangeDislaimerChecked() {
|
||||||
|
this.isDisclaimerChecked = !this.isDisclaimerChecked;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,48 +1,93 @@
|
|||||||
<h1 mat-dialog-title>
|
<h1 mat-dialog-title>
|
||||||
<span i18n>Create Account</span>
|
<span i18n>Create Account</span>
|
||||||
@if (data.role === 'ADMIN') {
|
@if (role === 'ADMIN') {
|
||||||
<span class="badge badge-light ml-2">{{ data.role }}</span>
|
<span class="badge badge-light ml-2">{{ role }}</span>
|
||||||
}
|
}
|
||||||
</h1>
|
</h1>
|
||||||
<div class="py-3" mat-dialog-content>
|
<div class="px-0" mat-dialog-content>
|
||||||
<div>
|
<mat-stepper #stepper animationDuration="0ms" linear>
|
||||||
<mat-form-field appearance="outline" class="w-100">
|
<mat-step editable="false" [completed]="isDisclaimerChecked">
|
||||||
<mat-label i18n>Security Token</mat-label>
|
<ng-template i18n matStepLabel>Terms and Conditions</ng-template>
|
||||||
<textarea
|
<div class="pt-2">
|
||||||
cdkTextareaAutosize
|
<ng-container i18n
|
||||||
matInput
|
>Please keep your security token safe. If you lose it, you will not be
|
||||||
readonly
|
able to recover your account.</ng-container
|
||||||
type="text"
|
|
||||||
[(value)]="data.accessToken"
|
|
||||||
></textarea>
|
|
||||||
<div class="float-right mt-3">
|
|
||||||
<button
|
|
||||||
color="secondary"
|
|
||||||
mat-flat-button
|
|
||||||
[cdkCopyToClipboard]="data.accessToken"
|
|
||||||
(click)="enableAgreeButton()"
|
|
||||||
>
|
>
|
||||||
<ion-icon class="mr-1" name="copy-outline" /><span i18n
|
|
||||||
>Copy to clipboard</span
|
|
||||||
>
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
</mat-form-field>
|
<mat-checkbox
|
||||||
</div>
|
class="pt-2"
|
||||||
<p i18n>
|
color="primary"
|
||||||
I agree to have stored my <i>Security Token</i> from above in a secure
|
(change)="onChangeDislaimerChecked()"
|
||||||
place. If I lose it, I cannot get my account back.
|
>
|
||||||
</p>
|
<ng-container i18n
|
||||||
</div>
|
>I understand that if I lose my security token, I cannot recover my
|
||||||
<div class="justify-content-end" mat-dialog-actions>
|
account.</ng-container
|
||||||
<button i18n mat-flat-button [mat-dialog-close]="undefined">Cancel</button>
|
>
|
||||||
<button
|
</mat-checkbox>
|
||||||
color="primary"
|
<div class="mt-3" mat-dialog-actions>
|
||||||
mat-flat-button
|
<div class="flex-grow-1">
|
||||||
[disabled]="isAgreeButtonDisabled"
|
<button i18n mat-button [mat-dialog-close]="undefined">Cancel</button>
|
||||||
[mat-dialog-close]="data"
|
</div>
|
||||||
>
|
<div>
|
||||||
<span i18n>Agree and continue</span>
|
<button
|
||||||
<ion-icon class="ml-1" name="arrow-forward-outline" />
|
color="primary"
|
||||||
</button>
|
mat-flat-button
|
||||||
|
[disabled]="!isDisclaimerChecked"
|
||||||
|
(click)="createAccount()"
|
||||||
|
>
|
||||||
|
<ng-container i18n>Continue</ng-container>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step editable="false">
|
||||||
|
<ng-template i18n matStepLabel>Security Token</ng-template>
|
||||||
|
<div class="pt-2">
|
||||||
|
<ng-container i18n
|
||||||
|
>Here is your security token. It is only visible once, please store
|
||||||
|
and keep it in a safe place.</ng-container
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<mat-form-field appearance="outline" class="pt-3 w-100 without-hint">
|
||||||
|
<mat-label i18n>Security Token</mat-label>
|
||||||
|
<textarea
|
||||||
|
cdkTextareaAutosize
|
||||||
|
matInput
|
||||||
|
readonly
|
||||||
|
type="text"
|
||||||
|
[(value)]="accessToken"
|
||||||
|
></textarea>
|
||||||
|
<div class="float-right mt-1">
|
||||||
|
<button
|
||||||
|
color="secondary"
|
||||||
|
mat-flat-button
|
||||||
|
[cdkCopyToClipboard]="accessToken"
|
||||||
|
(click)="enableCreateAccountButton()"
|
||||||
|
>
|
||||||
|
<ion-icon class="mr-1" name="copy-outline" />
|
||||||
|
<span i18n>Copy to clipboard</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</mat-form-field>
|
||||||
|
<div align="end" class="mt-1" mat-dialog-actions>
|
||||||
|
<div>
|
||||||
|
<button
|
||||||
|
color="primary"
|
||||||
|
mat-flat-button
|
||||||
|
matStepperNext
|
||||||
|
[disabled]="isCreateAccountButtonDisabled"
|
||||||
|
[mat-dialog-close]="authToken"
|
||||||
|
>
|
||||||
|
<span i18n>Create Account</span>
|
||||||
|
<ion-icon class="ml-1" name="arrow-forward-outline" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
|
||||||
|
<ng-template matStepperIcon="done">
|
||||||
|
<ion-icon name="checkmark-outline"></ion-icon>
|
||||||
|
</ng-template>
|
||||||
|
</mat-stepper>
|
||||||
|
<div></div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -4,9 +4,11 @@ import { CommonModule } from '@angular/common';
|
|||||||
import { CUSTOM_ELEMENTS_SCHEMA, NgModule } from '@angular/core';
|
import { CUSTOM_ELEMENTS_SCHEMA, NgModule } from '@angular/core';
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
||||||
import { MatButtonModule } from '@angular/material/button';
|
import { MatButtonModule } from '@angular/material/button';
|
||||||
|
import { MatCheckboxModule } from '@angular/material/checkbox';
|
||||||
import { MatDialogModule } from '@angular/material/dialog';
|
import { MatDialogModule } from '@angular/material/dialog';
|
||||||
import { MatFormFieldModule } from '@angular/material/form-field';
|
import { MatFormFieldModule } from '@angular/material/form-field';
|
||||||
import { MatInputModule } from '@angular/material/input';
|
import { MatInputModule } from '@angular/material/input';
|
||||||
|
import { MatStepperModule } from '@angular/material/stepper';
|
||||||
|
|
||||||
import { ShowAccessTokenDialog } from './show-access-token-dialog.component';
|
import { ShowAccessTokenDialog } from './show-access-token-dialog.component';
|
||||||
|
|
||||||
@ -17,9 +19,11 @@ import { ShowAccessTokenDialog } from './show-access-token-dialog.component';
|
|||||||
CommonModule,
|
CommonModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
MatButtonModule,
|
MatButtonModule,
|
||||||
|
MatCheckboxModule,
|
||||||
MatDialogModule,
|
MatDialogModule,
|
||||||
MatFormFieldModule,
|
MatFormFieldModule,
|
||||||
MatInputModule,
|
MatInputModule,
|
||||||
|
MatStepperModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
TextFieldModule
|
TextFieldModule
|
||||||
],
|
],
|
||||||
|
@ -1,2 +1,6 @@
|
|||||||
:host {
|
:host {
|
||||||
|
.mat-mdc-dialog-actions {
|
||||||
|
padding-left: 0 !important;
|
||||||
|
padding-right: 0 !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user