From 2c6b0c582a7ae1252934a95d37077ab863cc2457 Mon Sep 17 00:00:00 2001
From: Alex ORLUC <alex.orluc@maarch.org>
Date: Thu, 6 May 2021 16:12:25 +0200
Subject: [PATCH] FEAT #16982 TIME 3 WIP otp user creation

---
 lang/fr.json                                  |   8 +++--
 .../otps/otp-create.component.html            |   5 +++
 .../otps/otp-create.component.scss            |   7 +++-
 .../visa-workflow/otps/otp.service.ts         |  34 ++++++++++++++++++
 .../otps/yousign/otp-yousign.component.html   |  23 ++++++------
 .../otps/yousign/otp-yousign.component.ts     |  19 +++++++---
 .../visa-workflow.component.html              |  11 +++---
 .../visa-workflow/visa-workflow.component.ts  |   8 +++--
 .../app/indexation/indexation.component.ts    |   1 +
 src/frontend/app/service/auth.service.ts      |  11 +++++-
 src/frontend/assets/yousign.png               | Bin 0 -> 4763 bytes
 11 files changed, 99 insertions(+), 28 deletions(-)
 create mode 100644 src/frontend/app/document/visa-workflow/otps/otp.service.ts
 create mode 100644 src/frontend/assets/yousign.png

diff --git a/lang/fr.json b/lang/fr.json
index 172d2c6c36..23fc764fe6 100755
--- a/lang/fr.json
+++ b/lang/fr.json
@@ -476,9 +476,13 @@
 		"receiveActivationNotification": "Recevoir le courriel d'activation",
 		"newOtp": "Ajouter un OTP",
 		"otpUser": "Utilisateur OTP",
-		"notifyUserBy": "Notifier l'utilisateur par",
+		"securityCodeSendMode": "Mode d'envoi du code de sécurité",
 		"phoneAlt": "Mobile",
 		"sms": "Sms",
-		"source": "Source"
+		"source": "Source",
+		"otp_visa_yousignUser": "viseur (yousign)",
+		"otp_sign_yousignUser": "signataire (yousign)",
+		"role": "Role",
+		"otpMsg": "L'utilisateur sera notifié par <b>mail</b> et recevra un <b>code de sécurité</b> par <b>{{security}}</b> au moment de son tour dans le circuit."
 	}
 }
diff --git a/src/frontend/app/document/visa-workflow/otps/otp-create.component.html b/src/frontend/app/document/visa-workflow/otps/otp-create.component.html
index 66d8c5cf5a..3d7f1e33c5 100644
--- a/src/frontend/app/document/visa-workflow/otps/otp-create.component.html
+++ b/src/frontend/app/document/visa-workflow/otps/otp-create.component.html
@@ -9,6 +9,11 @@
     </ion-toolbar>
 </ion-header>
 <ion-content>
+    <ion-card>
+        <ion-item color="primary">
+            <ion-label class="info" [innerHTML]="'lang.otpMsg' | translate : { security : appOtpYousign?.getSecurityMode()}"></ion-label>
+        </ion-item>
+    </ion-card>
     <ion-card>
         <ion-item>
             <ion-label color="secondary">{{'lang.source' | translate}}</ion-label>
diff --git a/src/frontend/app/document/visa-workflow/otps/otp-create.component.scss b/src/frontend/app/document/visa-workflow/otps/otp-create.component.scss
index b6dd7ffb77..806b07f733 100644
--- a/src/frontend/app/document/visa-workflow/otps/otp-create.component.scss
+++ b/src/frontend/app/document/visa-workflow/otps/otp-create.component.scss
@@ -1,3 +1,8 @@
 .my-custom-class {
     --min-width: 400px;
-  }
\ No newline at end of file
+}
+
+.info {
+  white-space: initial;
+  line-height: 24px;
+}
\ No newline at end of file
diff --git a/src/frontend/app/document/visa-workflow/otps/otp.service.ts b/src/frontend/app/document/visa-workflow/otps/otp.service.ts
new file mode 100644
index 0000000000..59741d7e79
--- /dev/null
+++ b/src/frontend/app/document/visa-workflow/otps/otp.service.ts
@@ -0,0 +1,34 @@
+import { Injectable } from '@angular/core';
+import { HttpClient } from '@angular/common/http';
+import { catchError, tap } from 'rxjs/operators';
+import { of } from 'rxjs';
+import { NotificationService } from '../../../service/notification.service';
+
+@Injectable({
+    providedIn: 'root'
+})
+export class OtpService {
+
+    constructor(
+        public http: HttpClient,
+        public notificationService: NotificationService,
+    ) { }
+
+    getUserOtpIcon(id: string) {
+        return new Promise((resolve) => {
+            this.http.get(`assets/${id}.png`, { responseType: 'blob' }).pipe(
+                tap((response: any) => {
+                    const reader = new FileReader();
+                    reader.readAsDataURL(response);
+                    reader.onloadend = () => {
+                        resolve(reader.result as any);
+                    };
+                }),
+                catchError(err => {
+                    this.notificationService.handleErrors(err);
+                    return of(false);
+                })
+            ).subscribe();
+        });
+    }
+}
diff --git a/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.html b/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.html
index 8d37833aff..979fb5fb5c 100644
--- a/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.html
+++ b/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.html
@@ -14,26 +14,25 @@
             <ion-label color="secondary" position="floating">{{'lang.phoneAlt' | translate}} <ng-container
                     *ngIf="otp.mode === 'sms'">*</ng-container>
             </ion-label>
-            <ion-input name="phone" pattern="^\+?[1-9]\d{1,14}$" [(ngModel)]="otp.phone" placeholder="+33646342143"
+            <ion-input name="phone" pattern="^\+?[1-9]\d{1,14}$" [(ngModel)]="otp.phone" placeholder="+33646342143" (keyup)="formatPhone($event)"
                 [required]="otp.mode === 'sms'"></ion-input>
         </ion-item>
         <ion-item>
-            <ion-label color="secondary" position="floating">{{'lang.email' | translate}} <ng-container
-                    *ngIf="otp.mode === 'email'">*</ng-container>
+            <ion-label color="secondary" position="floating">{{'lang.email' | translate}} *
             </ion-label>
             <ion-input name="email" pattern="(^[a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9-.]+$)"
-                placeholder="alain.dupont@gmail.com" [(ngModel)]="otp.email" [required]="otp.mode === 'email'">
+                placeholder="alain.dupont@gmail.com" [(ngModel)]="otp.email" required>
             </ion-input>
         </ion-item>
     </ion-list>
     <ion-list>
-        <ion-radio-group [(ngModel)]="otp.role" [value]="otp.role">
+        <ion-radio-group name="role" [(ngModel)]="otp.role" [value]="otp.role">
             <ion-list-header>
-                <ion-label color="secondary">{{'lang.role' | translate}}</ion-label>
+                <ion-label color="secondary">{{'lang.role' | translate}} *</ion-label>
             </ion-list-header>
             <div style="display:flex;">
                 <ion-item *ngFor="let role of roles" style="flex:1;">
-                    <ion-label>{{'lang.' + role | translate}}</ion-label>
+                    <ion-label>{{'lang.' + role + 'User' | translate}}</ion-label>
                     <ion-radio slot="start" [value]="role"></ion-radio>
                 </ion-item>
             </div>
@@ -46,11 +45,11 @@
     </ion-list>
     <ion-list>
         <ion-item>
-            <ion-label color="secondary" position="floating">{{'lang.notifyUserBy' | translate}}</ion-label>
-            <ion-select name="mode" [(ngModel)]="otp.notify" [value]="otp.notify"
-                cancelText="{{'lang.cancel' | translate}}" [disabled]="notificationModes.length === 1">
-                <ion-select-option *ngFor="let mode of notificationModes" [value]="mode">
-                    {{'lang.' + mode | translate}}</ion-select-option>
+            <ion-label color="secondary" position="floating">{{'lang.securityCodeSendMode' | translate}} *</ion-label>
+            <ion-select name="mode" [(ngModel)]="otp.security" [value]="otp.security"
+                cancelText="{{'lang.cancel' | translate}}" [disabled]="securityModes.length === 1">
+                <ion-select-option *ngFor="let security of securityModes" [value]="security">
+                    {{'lang.' + security | translate}}</ion-select-option>
             </ion-select>
         </ion-item>
     </ion-list>
diff --git a/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.ts b/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.ts
index 48f33845f2..dd62f5be4e 100644
--- a/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.ts
+++ b/src/frontend/app/document/visa-workflow/otps/yousign/otp-yousign.component.ts
@@ -12,7 +12,7 @@ import { NgForm } from '@angular/forms';
 export class OtpYousignComponent implements OnInit {
     @ViewChild('otpForm', { static: false }) otpForm: NgForm;
 
-    notificationModes: any[] = [];
+    securityModes: any[] = [];
 
     roles: any[] = [
         'otp_visa_yousign',
@@ -20,11 +20,12 @@ export class OtpYousignComponent implements OnInit {
     ];
 
     otp: any = {
+        type: 'yousign',
         firstname: '',
         lastname: '',
         email: '',
         phone: '',
-        notify: 'sms',
+        security: 'sms',
         role: 'otp_sign_yousign'
     };
 
@@ -40,11 +41,11 @@ export class OtpYousignComponent implements OnInit {
 
     getConfig() {
         // FOR TEST
-        this.notificationModes = [
+        this.securityModes = [
             'sms',
             'email'
         ];
-        this.otp.mode = this.notificationModes[0];
+        this.otp.security = this.securityModes[0];
         /* this.http.get(`../rest/???`).pipe(
             tap((data: any) => {
 
@@ -60,7 +61,17 @@ export class OtpYousignComponent implements OnInit {
         return this.otp;
     }
 
+    getSecurityMode() {
+        return this.translate.instant('lang.' + this.otp.security);
+    }
+
     isValid() {
         return this.otpForm.valid;
     }
+
+    formatPhone(ev: any) {
+        if (this.otp.phone.length > 1 && this.otp.phone[0] === '0') {
+            this.otp.phone = this.otp.phone.replace('0', '+33');
+        }
+    }
 }
diff --git a/src/frontend/app/document/visa-workflow/visa-workflow.component.html b/src/frontend/app/document/visa-workflow/visa-workflow.component.html
index b3f01cf772..aaf9961e88 100644
--- a/src/frontend/app/document/visa-workflow/visa-workflow.component.html
+++ b/src/frontend/app/document/visa-workflow/visa-workflow.component.html
@@ -14,22 +14,21 @@
             <ion-searchbar #searchInput [(ngModel)]="visaUsersSearchVal" [placeholder]="'lang.searchUser' | translate"
                 (ionChange)="getVisaUsers($event)" (ionFocus)="visaUsersSearchVal=''"></ion-searchbar>
             <ion-buttons slot="end">
-                <ion-button fill="clear" slot="icon-only" shape="round" color="primary" [matMenuTriggerFor]="menu"
-                    [title]="'lang.circuitModels' | translate">
-                    <ion-icon slot="icon-only" name="albums-outline"></ion-icon>
+                <ion-button fill="clear" slot="icon-only" shape="round" color="primary" [matMenuTriggerFor]="menu">
+                    <ion-icon slot="icon-only" name="ellipsis-vertical-outline"></ion-icon>
                 </ion-button>
             </ion-buttons>
             <mat-menu #menu="matMenu">
                 <ion-item button (click)="openOtpModal()" lines="none">
                     <ion-icon name="person-circle-outline" slot="start" color="primary"></ion-icon>
-                    <ion-label>{{'lang.newOtp' | translate}}</ion-label>
+                    <ion-label style="font-size: 14px;">{{'lang.newOtp' | translate}}</ion-label>
                 </ion-item>
                 <button mat-menu-item [matMenuTriggerFor]="model"
                     (menuOpened)="getVisaUserModels()">{{'lang.circuitModels' | translate}}</button>
             </mat-menu>
             <mat-menu #model="matMenu">
                 <ion-item button lines="none" *ngFor="let model of visaWorkflowModels" (click)="loadVisaWorkflow(model)">
-                    <ion-label>{{model.title}}</ion-label>
+                    <ion-label style="font-size: 14px;">{{model.title}}</ion-label>
                     <ion-buttons slot="end">
                         <ion-button fill="clear" slot="icon-only" shape="round" color="danger"
                             (click)="$event.stopPropagation();removeModel(model)">
@@ -38,7 +37,7 @@
                     </ion-buttons>
                 </ion-item>
                 <ion-item button (click)="createModel()" [disabled]="visaWorkflow.length === 0">
-                    <ion-label>{{'lang.newTemplate' | translate}}</ion-label>
+                    <ion-label style="font-size: 14px;">{{'lang.newTemplate' | translate}}</ion-label>
                 </ion-item>
             </mat-menu>
         </ion-item>
diff --git a/src/frontend/app/document/visa-workflow/visa-workflow.component.ts b/src/frontend/app/document/visa-workflow/visa-workflow.component.ts
index a99c7ec7d4..45299a66c2 100644
--- a/src/frontend/app/document/visa-workflow/visa-workflow.component.ts
+++ b/src/frontend/app/document/visa-workflow/visa-workflow.component.ts
@@ -9,6 +9,7 @@ import { AlertController, IonReorderGroup, ModalController, PopoverController }
 import { ItemReorderEventDetail } from '@ionic/core';
 import { TranslateService } from '@ngx-translate/core';
 import { OtpCreateComponent } from './otps/otp-create.component';
+import { OtpService } from './otps/otp.service';
 
 @Component({
     selector: 'app-visa-workflow',
@@ -40,6 +41,7 @@ export class VisaWorkflowComponent implements OnInit {
         public authService: AuthService,
         public notificationService: NotificationService,
         public modalController: ModalController,
+        private otpService: OtpService,
     ) { }
 
     ngOnInit(): void {
@@ -156,15 +158,17 @@ export class VisaWorkflowComponent implements OnInit {
         await modal.present();
 
         modal.onDidDismiss()
-            .then((result: any) => {
+            .then(async (result: any) => {
                 if (typeof result.data === 'object') {
                     const obj: any = {
                         'userId': null,
                         'userDisplay': `${result.data.firstname} ${result.data.lastname}`,
+                        'userPicture': await this.otpService.getUserOtpIcon(result.data.type),
                         'role': result.data.role,
                         'processDate': null,
                         'current': false,
-                        'modes': [result.data.role]
+                        'modes': [result.data.role],
+                        'otp': result.data
                     };
                     this.visaWorkflow.push(obj);
                 }
diff --git a/src/frontend/app/indexation/indexation.component.ts b/src/frontend/app/indexation/indexation.component.ts
index 1f2e755458..1fd81100dd 100644
--- a/src/frontend/app/indexation/indexation.component.ts
+++ b/src/frontend/app/indexation/indexation.component.ts
@@ -255,6 +255,7 @@ export class IndexationComponent implements OnInit {
                     encodedDocument: item.content
                 })),
                 workflow: this.appVisaWorkflow.getCurrentWorkflow().map((item: any, index: number) => ({
+                    externalInformations: item.otp,
                     userId: item.userId,
                     mode: this.authService.getWorkflowMode(item.role),
                     signatureMode: this.authService.getSignatureMode(item.role),
diff --git a/src/frontend/app/service/auth.service.ts b/src/frontend/app/service/auth.service.ts
index c6c920d0d2..3b8616bcda 100755
--- a/src/frontend/app/service/auth.service.ts
+++ b/src/frontend/app/service/auth.service.ts
@@ -137,7 +137,16 @@ export class AuthService {
     }
 
     getWorkflowMode(id: string) {
-        return this.signatureRoles.filter((item: any) => item.id === id)[0].type;
+        const type = this.signatureRoles.filter((item: any) => item.id === id)[0]?.type;
+        const isOtp = /otp_[.]*/g;
+        const isSign = /_sign_/g;
+
+        // OTP user
+        if (id.match(isOtp) !== null) {
+            return id.match(isSign) !== null ? 'sign' : 'sign';
+        } else {
+            return type;
+        }
     }
 
     setCachedUrl(url: string) {
diff --git a/src/frontend/assets/yousign.png b/src/frontend/assets/yousign.png
new file mode 100644
index 0000000000000000000000000000000000000000..d308e1de7bb85b2ec6c9ca2ff34f6739dafb3d47
GIT binary patch
literal 4763
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

literal 0
HcmV?d00001

-- 
GitLab