@universis/common
Version:
Universis - common directives and services
168 lines • 41.3 kB
JavaScript
import { Component, Input } from '@angular/core';
import { CertificateService } from '../../services/certificate-service/certificate-service.service';
import { BsModalRef } from 'ngx-bootstrap/modal';
import { TranslateService } from '@ngx-translate/core';
import { environment } from '../../../environments/environment';
import { SIGNATURE_INFO } from './i18n/signature-info.translations';
import * as i0 from "@angular/core";
import * as i1 from "ngx-bootstrap/modal";
import * as i2 from "../../services/certificate-service/certificate-service.service";
import * as i3 from "@ngx-translate/core";
import * as i4 from "@angular/common";
export class SignatureInfoComponent {
constructor(_bsModalRef, _certificateService, _translateService) {
this._bsModalRef = _bsModalRef;
this._certificateService = _certificateService;
this._translateService = _translateService;
/**
* The list of attributes to show to the user
*/
this.signatureAttributes = [];
this.tab = 1;
this.generalAttributes = [];
this.purposes = [];
environment.languages.forEach((culture) => {
if (SIGNATURE_INFO[culture])
this._translateService.setTranslation(culture, SIGNATURE_INFO[culture], true);
});
}
async ngOnInit() {
this.signatureAttributes = [
{
translationKey: 'SignatureInfo.CheckHash',
value: this.checkHashKey
},
{
translationKey: 'SignatureInfo.SignatureBlock',
value: this.signatureBlock
}
];
try {
let parsed;
parsed = this._certificateService.getX509Certificate(this.certificate);
if (!this.user) {
this.user = this._certificateService.extractCertificateOwner(parsed);
}
const certAttributes = this.resolveAttributes(parsed);
this.purposes = this._certificateService.extractPurposes(parsed);
this.signatureAttributes = [...certAttributes, ...this.signatureAttributes];
this.generalAttributes = this.getGeneralAttributes(parsed);
}
catch (err) {
console.log('err: ', err);
}
}
/**
*
* Resolves a certificate to usable fields
*
* @param {X509} certificate The X509 object certificate
*
*/
resolveAttributes(certificate) {
const certificateAttributes = [];
certificateAttributes.push({
translationKey: 'SignatureInfo.Version',
value: certificate.getVersion()
});
certificateAttributes.push({
translationKey: 'SignatureInfo.SerialNumber',
value: certificate.getSerialNumberHex()
});
certificateAttributes.push({
translationKey: 'SignatureInfo.Algorithm',
value: certificate.getSignatureAlgorithmField()
});
certificateAttributes.push({
translationKey: 'SignatureInfo.Issuer',
value: (certificate.getIssuer().str).replace(/\//g, '\n')
});
const notBefore = this._certificateService.parseCertificateDate(certificate.getNotBefore());
certificateAttributes.push({
translationKey: 'SignatureInfo.NotValidBefore',
value: notBefore,
validation: {
isValid: notBefore ? new Date() >= notBefore : false,
showValidation: true
}
});
const notAfter = this._certificateService.parseCertificateDate(certificate.getNotAfter());
certificateAttributes.push({
translationKey: 'SignatureInfo.NotValidAfter',
value: notAfter,
validation: {
isValid: notAfter ? notAfter >= new Date() : false,
showValidation: true
}
});
certificateAttributes.push({
translationKey: 'SignatureInfo.Subject',
value: (certificate.getSubject().str).replace(/\//g, '\n')
});
certificateAttributes.push({
translationKey: 'SignatureInfo.PublicKey',
value: certificate.getPublicKeyHex()
});
certificateAttributes.push({
translationKey: 'SignatureInfo.PublicKeyParameters',
value: certificate.getPublicKeyIdx()
});
return certificateAttributes;
}
getGeneralAttributes(certificate) {
let attributes = [];
attributes.push({
translationKey: 'SignatureInfo.IssuedTo',
// get certificate subject common name
value: (certificate.getSubjectString()).split('/').filter(x => x.includes('CN')).join(',').split('=')[1]
});
attributes.push({
translationKey: 'SignatureInfo.Issuer',
// get certificate issuer common name (CN)
value: (certificate.getIssuerString()).split('/').filter(x => x.includes('CN')).join(',').split('=')[1]
});
const notBefore = this._certificateService.parseCertificateDate(certificate.getNotBefore());
attributes.push({
translationKey: 'SignatureInfo.NotValidBeforeShort',
value: notBefore,
validation: {
isValid: notBefore ? new Date() >= notBefore : false,
showValidation: true
}
});
const notAfter = this._certificateService.parseCertificateDate(certificate.getNotAfter());
attributes.push({
translationKey: 'SignatureInfo.NotValidAfterShort',
value: notAfter,
validation: {
isValid: notAfter ? notAfter >= new Date() : false,
showValidation: true
}
});
return attributes;
}
hideCertificateModal() {
this._bsModalRef.hide();
}
}
SignatureInfoComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: SignatureInfoComponent, deps: [{ token: i1.BsModalRef }, { token: i2.CertificateService }, { token: i3.TranslateService }], target: i0.ɵɵFactoryTarget.Component });
SignatureInfoComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: SignatureInfoComponent, selector: "universis-signature-info", inputs: { user: "user", isVerified: "isVerified", certificate: "certificate", signatureBlock: "signatureBlock", checkHashKey: "checkHashKey", examinationCode: "examinationCode", dateCreated: "dateCreated" }, ngImport: i0, template: "<div class=\"modal-header p-0\">\n <button\n (click)=\"hideCertificateModal()\"\n type=\"button\"\n class=\"close\"\n data-dismiss=\"modal\"\n aria-label=\"Cancel\"\n >\n <span aria-hidden=\"true\">×</span>\n </button>\n</div>\n<div class=\"modal-body mt-0 pt-0\">\n <div class=\"card-body\" *ngIf=\"user\">\n <div class=\"row\">\n <div class=\"col-12\">\n <h5 class=\"mt-2 mb-3\">\n {{user?.familyName}} {{user?.givenName}}\n <span *ngIf=\"dateCreated\">{{ dateCreated | date:'short' }}</span>\n (\n <span *ngIf=\"isVerified\" class=\"text-purple\" [translate]=\"'SignatureInfo.Verified'\"></span>\n <span *ngIf=\"!isVerified\" class=\"text-red\" [translate]=\"'SignatureInfo.NotVerified'\"></span>\n )\n <small> {{examinationCode}} </small>\n </h5>\n </div>\n </div>\n <div class=\"mb-3\">\n <div class=\" d-none d-md-block\">\n <ul class=\"nav nav-tabs\">\n <li class=\"nav-item\">\n <a class=\"nav-link active\" [class.active]=\"tab === 1\" rel=\"tab1\"\n (click)=\"tab =1\">{{'SignatureInfo.General' | translate}}</a>\n </li>\n <li class=\"nav-item\">\n <a class=\"nav-link\" [class.active]=\"tab === 2\" rel=\"tab2\"\n (click)=\"tab =2\"> {{'SignatureInfo.Details' | translate}}</a>\n </li>\n </ul>\n </div>\n </div>\n\n <div *ngIf=\"tab===1\">\n <div class=\"card\">\n <h5 class=\"d-block card-header mt-2 mb-n2 align-items-center\">\n <i class=\"fa fa-certificate\"></i>\n {{'SignatureInfo.CertificateInformation' | translate}}\n </h5>\n <hr/>\n <div class=\"d-block card-body\">\n <div class=\"container font-lg mt-n2 pt-n2\" *ngIf=\"purposes && purposes.length\">\n <div class=\"row\">\n <div>\n <span> {{'SignatureInfo.CertificatePurposes.Header' |translate}}</span>\n </div>\n </div>\n <ul>\n <li *ngFor=\"let purpose of purposes\">\n {{'SignatureInfo.CertificatePurposes.' + purpose |translate}}\n </li>\n </ul>\n <hr/>\n </div>\n <div *ngFor=\"let attribute of generalAttributes | slice: 0:2\" class=\"row font-lg\">\n <div *ngIf=\"attribute && attribute.value\">\n <div class=\"flex-sm-column flex-wrap\">\n <span class=\"text-theme col-4 text-nowrap\">{{attribute.translationKey | translate}}: </span>\n <span class=\"col-8\">{{attribute.value}}</span>\n </div>\n </div>\n </div>\n <div class=\"row font-lg\">\n <span *ngFor=\"let attribute of generalAttributes |slice: 2:4\" class=\"flex-sm-column\">\n <span *ngIf=\"attribute && attribute.value\">\n <span class=\"col-4 text-theme text-nowrap\">{{attribute.translationKey | translate}}: </span>\n <span class=\"col-4 pr-2\">{{attribute.value |date:'short'}}</span>\n </span>\n </span>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"tab=== 2\">\n <div id=\"group-list\" class=\"flex-column\">\n <div class=\"sis--list-group sis--list-group-body\">\n <ul class=\"sis--list-group sis--list-group-header\">\n <li class=\"sis--list-group-item\">\n <div class=\"d-flex justify-content-center align-items-center\">\n <div class=\"sis--list-group-item_lg_column d-none d-block\"\n [translate]=\"'SignatureInfo.Field'\"></div>\n <div class=\"sis--list-group-item_lg_column d-none d-block px-4\"\n [translate]=\"'SignatureInfo.Value'\"></div>\n </div>\n </li>\n </ul>\n <ul class=\"sis--list-group sis--list-group-body\">\n <li class=\"sis--list-group-item\" *ngFor=\"let attribute of signatureAttributes\">\n <div class=\"d-flex justify-content-center align-items-center list-color\">\n <div class=\"sis--list-group-item_lg_column d-none d-block flex-wrap\">\n <span class=\"text-secondary\">\n <i class=\"fa fa-certificate\"></i>\n {{attribute.translationKey | translate}}\n </span>\n </div>\n <div class=\"sis--list-group-item_lg_column d-block flex-wrap\">\n <span class=\"text-secondary p-2 p-md-0 text-break \" [ngClass]=\"{\n 'text-red': attribute.validation && attribute.validation.showValidation && !attribute.validation.isValid,\n 'text-green': attribute.validation && attribute.validation.showValidation && attribute.validation.isValid\n }\">{{ attribute.value}}</span>\n </div>\n </div>\n </li>\n </ul>\n </div>\n </div>\n </div>\n </div>\n <div class=\"row\">\n <div class=\"col text-center\">\n <button\n (click)=\"hideCertificateModal()\"\n class=\"btn btn-success\"\n [translate]=\"'SignatureInfo.Close'\"\n ></button>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "pipe", type: i4.SlicePipe, name: "slice" }, { kind: "pipe", type: i4.DatePipe, name: "date" }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: SignatureInfoComponent, decorators: [{
type: Component,
args: [{ selector: 'universis-signature-info', template: "<div class=\"modal-header p-0\">\n <button\n (click)=\"hideCertificateModal()\"\n type=\"button\"\n class=\"close\"\n data-dismiss=\"modal\"\n aria-label=\"Cancel\"\n >\n <span aria-hidden=\"true\">×</span>\n </button>\n</div>\n<div class=\"modal-body mt-0 pt-0\">\n <div class=\"card-body\" *ngIf=\"user\">\n <div class=\"row\">\n <div class=\"col-12\">\n <h5 class=\"mt-2 mb-3\">\n {{user?.familyName}} {{user?.givenName}}\n <span *ngIf=\"dateCreated\">{{ dateCreated | date:'short' }}</span>\n (\n <span *ngIf=\"isVerified\" class=\"text-purple\" [translate]=\"'SignatureInfo.Verified'\"></span>\n <span *ngIf=\"!isVerified\" class=\"text-red\" [translate]=\"'SignatureInfo.NotVerified'\"></span>\n )\n <small> {{examinationCode}} </small>\n </h5>\n </div>\n </div>\n <div class=\"mb-3\">\n <div class=\" d-none d-md-block\">\n <ul class=\"nav nav-tabs\">\n <li class=\"nav-item\">\n <a class=\"nav-link active\" [class.active]=\"tab === 1\" rel=\"tab1\"\n (click)=\"tab =1\">{{'SignatureInfo.General' | translate}}</a>\n </li>\n <li class=\"nav-item\">\n <a class=\"nav-link\" [class.active]=\"tab === 2\" rel=\"tab2\"\n (click)=\"tab =2\"> {{'SignatureInfo.Details' | translate}}</a>\n </li>\n </ul>\n </div>\n </div>\n\n <div *ngIf=\"tab===1\">\n <div class=\"card\">\n <h5 class=\"d-block card-header mt-2 mb-n2 align-items-center\">\n <i class=\"fa fa-certificate\"></i>\n {{'SignatureInfo.CertificateInformation' | translate}}\n </h5>\n <hr/>\n <div class=\"d-block card-body\">\n <div class=\"container font-lg mt-n2 pt-n2\" *ngIf=\"purposes && purposes.length\">\n <div class=\"row\">\n <div>\n <span> {{'SignatureInfo.CertificatePurposes.Header' |translate}}</span>\n </div>\n </div>\n <ul>\n <li *ngFor=\"let purpose of purposes\">\n {{'SignatureInfo.CertificatePurposes.' + purpose |translate}}\n </li>\n </ul>\n <hr/>\n </div>\n <div *ngFor=\"let attribute of generalAttributes | slice: 0:2\" class=\"row font-lg\">\n <div *ngIf=\"attribute && attribute.value\">\n <div class=\"flex-sm-column flex-wrap\">\n <span class=\"text-theme col-4 text-nowrap\">{{attribute.translationKey | translate}}: </span>\n <span class=\"col-8\">{{attribute.value}}</span>\n </div>\n </div>\n </div>\n <div class=\"row font-lg\">\n <span *ngFor=\"let attribute of generalAttributes |slice: 2:4\" class=\"flex-sm-column\">\n <span *ngIf=\"attribute && attribute.value\">\n <span class=\"col-4 text-theme text-nowrap\">{{attribute.translationKey | translate}}: </span>\n <span class=\"col-4 pr-2\">{{attribute.value |date:'short'}}</span>\n </span>\n </span>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"tab=== 2\">\n <div id=\"group-list\" class=\"flex-column\">\n <div class=\"sis--list-group sis--list-group-body\">\n <ul class=\"sis--list-group sis--list-group-header\">\n <li class=\"sis--list-group-item\">\n <div class=\"d-flex justify-content-center align-items-center\">\n <div class=\"sis--list-group-item_lg_column d-none d-block\"\n [translate]=\"'SignatureInfo.Field'\"></div>\n <div class=\"sis--list-group-item_lg_column d-none d-block px-4\"\n [translate]=\"'SignatureInfo.Value'\"></div>\n </div>\n </li>\n </ul>\n <ul class=\"sis--list-group sis--list-group-body\">\n <li class=\"sis--list-group-item\" *ngFor=\"let attribute of signatureAttributes\">\n <div class=\"d-flex justify-content-center align-items-center list-color\">\n <div class=\"sis--list-group-item_lg_column d-none d-block flex-wrap\">\n <span class=\"text-secondary\">\n <i class=\"fa fa-certificate\"></i>\n {{attribute.translationKey | translate}}\n </span>\n </div>\n <div class=\"sis--list-group-item_lg_column d-block flex-wrap\">\n <span class=\"text-secondary p-2 p-md-0 text-break \" [ngClass]=\"{\n 'text-red': attribute.validation && attribute.validation.showValidation && !attribute.validation.isValid,\n 'text-green': attribute.validation && attribute.validation.showValidation && attribute.validation.isValid\n }\">{{ attribute.value}}</span>\n </div>\n </div>\n </li>\n </ul>\n </div>\n </div>\n </div>\n </div>\n <div class=\"row\">\n <div class=\"col text-center\">\n <button\n (click)=\"hideCertificateModal()\"\n class=\"btn btn-success\"\n [translate]=\"'SignatureInfo.Close'\"\n ></button>\n </div>\n </div>\n</div>\n" }]
}], ctorParameters: function () { return [{ type: i1.BsModalRef }, { type: i2.CertificateService }, { type: i3.TranslateService }]; }, propDecorators: { user: [{
type: Input
}], isVerified: [{
type: Input
}], certificate: [{
type: Input
}], signatureBlock: [{
type: Input
}], checkHashKey: [{
type: Input
}], examinationCode: [{
type: Input
}], dateCreated: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signature-info.component.js","sourceRoot":"","sources":["../../../../../src/shared/components/signature-info/signature-info.component.ts","../../../../../src/shared/components/signature-info/signature-info.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAU,MAAM,eAAe,CAAC;AAEzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,gEAAgE,CAAC;AACpG,OAAO,EAAC,UAAU,EAAC,MAAM,qBAAqB,CAAC;AAC/C,OAAO,EAAC,gBAAgB,EAAC,MAAM,qBAAqB,CAAC;AACrD,OAAO,EAAC,WAAW,EAAC,MAAM,mCAAmC,CAAC;AAC9D,OAAO,EAAC,cAAc,EAAC,MAAM,oCAAoC,CAAC;;;;;;AAwBlE,MAAM,OAAO,sBAAsB;IA4BjC,YAAoB,WAAuB,EACvB,mBAAuC,EACvC,iBAAmC;QAFnC,gBAAW,GAAX,WAAW,CAAY;QACvB,wBAAmB,GAAnB,mBAAmB,CAAoB;QACvC,sBAAiB,GAAjB,iBAAiB,CAAkB;QAVvD;;WAEG;QACI,wBAAmB,GAAyB,EAAE,CAAC;QAC/C,QAAG,GAAY,CAAC,CAAC;QACjB,sBAAiB,GAAyB,EAAE,CAAC;QAC7C,aAAQ,GAAe,EAAE,CAAC;QAK/B,WAAW,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YACxC,IAAG,cAAc,CAAC,OAAO,CAAC;gBACxB,IAAI,CAAC,iBAAiB,CAAC,cAAc,CAAC,OAAO,EAAE,cAAc,CAAC,OAAO,CAAC,EAAE,IAAI,CAAC,CAAC;QAClF,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,mBAAmB,GAAG;YACzB;gBACE,cAAc,EAAE,yBAAyB;gBACzC,KAAK,EAAE,IAAI,CAAC,YAAY;aACzB;YACD;gBACE,cAAc,EAAE,8BAA8B;gBAC9C,KAAK,EAAE,IAAI,CAAC,cAAc;aAC3B;SACF,CAAC;QACF,IAAI;YACF,IAAI,MAAM,CAAC;YACX,MAAM,GAAG,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACvE,IAAG,CAAC,IAAI,CAAC,IAAI,EAAC;gBACZ,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,mBAAmB,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;aACtE;YACD,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;YACtD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;YACjE,IAAI,CAAC,mBAAmB,GAAG,CAAC,GAAG,cAAc,EAAE,GAAG,IAAI,CAAC,mBAAmB,CAAC,CAAC;YAC5E,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,CAAC;SAC5D;QAAC,OAAM,GAAG,EAAE;YACX,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;SAC3B;IACH,CAAC;IAED;;;;;;OAMG;IACH,iBAAiB,CAAC,WAAiB;QACjC,MAAM,qBAAqB,GAIrB,EAAE,CAAC;QACT,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,uBAAuB;YACvC,KAAK,EAAE,WAAW,CAAC,UAAU,EAAE;SAChC,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,4BAA4B;YAC5C,KAAK,EAAE,WAAW,CAAC,kBAAkB,EAAE;SACxC,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,yBAAyB;YACzC,KAAK,EAAE,WAAW,CAAC,0BAA0B,EAAE;SAChD,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,sBAAsB;YACtC,KAAK,EAAE,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC;SAC1D,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAC7D,WAAW,CAAC,YAAY,EAAE,CAC3B,CAAC;QAEF,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,8BAA8B;YAC9C,KAAK,EAAE,SAAS;YAChB,UAAU,EAAE;gBACV,OAAO,EAAE,SAAS,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,IAAI,SAAS,CAAC,CAAC,CAAC,KAAK;gBACpD,cAAc,EAAE,IAAI;aACrB;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAC5D,WAAW,CAAC,WAAW,EAAE,CAC1B,CAAA;QAED,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,6BAA6B;YAC7C,KAAK,EAAG,QAAQ;YAChB,UAAU,EAAE;gBACV,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,QAAQ,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK;gBAClD,cAAc,EAAE,IAAI;aACrB;SACF,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,uBAAuB;YACvC,KAAK,EAAE,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC;SAC3D,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,yBAAyB;YACzC,KAAK,EAAE,WAAW,CAAC,eAAe,EAAE;SACrC,CAAC,CAAC;QACH,qBAAqB,CAAC,IAAI,CAAC;YACzB,cAAc,EAAE,mCAAmC;YACnD,KAAK,EAAE,WAAW,CAAC,eAAe,EAAE;SACrC,CAAC,CAAC;QACH,OAAO,qBAAqB,CAAC;IAC/B,CAAC;IAED,oBAAoB,CAAC,WAAW;QAC9B,IAAI,UAAU,GAIR,EAAE,CAAC;QACT,UAAU,CAAC,IAAI,CAAC;YACd,cAAc,EAAE,wBAAwB;YACxC,sCAAsC;YACtC,KAAK,EAAE,CAAC,WAAW,CAAC,gBAAgB,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;SACzG,CAAC,CAAC;QACH,UAAU,CAAC,IAAI,CAAC;YACd,cAAc,EAAE,sBAAsB;YACtC,0CAA0C;YAC1C,KAAK,EAAE,CAAC,WAAW,CAAC,eAAe,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;SACxG,CAAC,CAAC;QACH,MAAM,SAAS,GAAG,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAC7D,WAAW,CAAC,YAAY,EAAE,CAC3B,CAAC;QACF,UAAU,CAAC,IAAI,CAAC;YACd,cAAc,EAAE,mCAAmC;YACnD,KAAK,EAAE,SAAS;YAChB,UAAU,EAAE;gBACV,OAAO,EAAE,SAAS,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,IAAI,SAAS,CAAC,CAAC,CAAC,KAAK;gBACpD,cAAc,EAAE,IAAI;aACrB;SACF,CAAC,CAAC;QACH,MAAM,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAC5D,WAAW,CAAC,WAAW,EAAE,CAC1B,CAAC;QACF,UAAU,CAAC,IAAI,CAAC;YACd,cAAc,EAAE,kCAAkC;YAClD,KAAK,EAAG,QAAQ;YAChB,UAAU,EAAE;gBACV,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,QAAQ,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK;gBAClD,cAAc,EAAE,IAAI;aACrB;SACF,CAAC,CAAC;QACH,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,oBAAoB;QAClB,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;IAC1B,CAAC;;oHAhLU,sBAAsB;wGAAtB,sBAAsB,gRC9BnC,isKA+HA;4FDjGa,sBAAsB;kBAJlC,SAAS;+BACE,0BAA0B;iKAQ3B,IAAI;sBAAZ,KAAK;gBAOG,UAAU;sBAAlB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACG,WAAW;sBAAnB,KAAK","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { X509 } from 'jsrsasign';\nimport { CertificateService } from '../../services/certificate-service/certificate-service.service';\nimport {BsModalRef} from 'ngx-bootstrap/modal';\nimport {TranslateService} from '@ngx-translate/core';\nimport {environment} from '../../../environments/environment';\nimport {SIGNATURE_INFO} from './i18n/signature-info.translations';\n\n/**\n *\n * @interface SignatureAttribute\n *\n * @property {string} translationKey The translation key to show to the user\n * @property {string} value The value to show to the user\n * @property {object} Validation data. isValid is the validation status, showValidation indicates whether to show the validation\n *\n */\ninterface SignatureAttribute {\n  translationKey: string;\n  value?: string;\n  validation?: {\n    isValid: boolean;\n    showValidation: boolean;\n  }\n}\n\n@Component({\n  selector: 'universis-signature-info',\n  templateUrl: './signature-info.component.html'\n})\nexport class SignatureInfoComponent implements OnInit {\n\n  /**\n   * @property {Object} user The user who submitted the exam  document\n   */\n  @Input() user;\n\n  /**\n   * @property {boolean} isVerified A flag to indicate whether the signature was\n   * verified at least once on server side. Can be used to show to the user that\n   * the signature was accepted at some point in the past even if the certificate is expired.\n   */\n  @Input() isVerified: boolean | undefined;\n  @Input() certificate: string | undefined;\n  @Input() signatureBlock: string | undefined;\n  @Input() checkHashKey: string | undefined;\n  @Input() examinationCode: string | undefined;\n  @Input() dateCreated: Date | undefined;\n\n\n  /**\n   * The list of attributes to show to the user\n   */\n  public signatureAttributes: SignatureAttribute[] = [];\n  public tab: number  = 1;\n  public generalAttributes: SignatureAttribute[] = [];\n  public purposes: Array<any> = [];\n\n  constructor(private _bsModalRef: BsModalRef,\n              private _certificateService: CertificateService,\n              private _translateService: TranslateService) {\n    environment.languages.forEach((culture) => {\n      if(SIGNATURE_INFO[culture])\n        this._translateService.setTranslation(culture, SIGNATURE_INFO[culture], true);\n    });\n  }\n\n  async ngOnInit() {\n    this.signatureAttributes = [\n      {\n        translationKey: 'SignatureInfo.CheckHash',\n        value: this.checkHashKey\n      },\n      {\n        translationKey: 'SignatureInfo.SignatureBlock',\n        value: this.signatureBlock\n      }\n    ];\n    try {\n      let parsed;\n      parsed = this._certificateService.getX509Certificate(this.certificate);\n      if(!this.user){\n        this.user = this._certificateService.extractCertificateOwner(parsed);\n      }\n      const certAttributes = this.resolveAttributes(parsed);\n      this.purposes = this._certificateService.extractPurposes(parsed);\n      this.signatureAttributes = [...certAttributes, ...this.signatureAttributes];\n      this.generalAttributes = this.getGeneralAttributes(parsed);\n    } catch(err) {\n      console.log('err: ', err);\n    }\n  }\n\n  /**\n   *\n   * Resolves a certificate to usable fields\n   *\n   * @param {X509} certificate The X509 object certificate\n   *\n   */\n  resolveAttributes(certificate: X509) {\n    const certificateAttributes: {\n      translationKey: string,\n      value?: any,\n      validation?: any\n    }[] = [];\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.Version',\n      value: certificate.getVersion()\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.SerialNumber',\n      value: certificate.getSerialNumberHex()\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.Algorithm',\n      value: certificate.getSignatureAlgorithmField()\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.Issuer',\n      value: (certificate.getIssuer().str).replace(/\\//g, '\\n')\n    });\n\n    const notBefore = this._certificateService.parseCertificateDate(\n      certificate.getNotBefore()\n    );\n\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.NotValidBefore',\n      value: notBefore,\n      validation: {\n        isValid: notBefore ? new Date() >= notBefore : false,\n        showValidation: true\n      }\n    });\n\n    const notAfter = this._certificateService.parseCertificateDate(\n      certificate.getNotAfter()\n    )\n\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.NotValidAfter',\n      value:  notAfter,\n      validation: {\n        isValid: notAfter ? notAfter >= new Date() : false,\n        showValidation: true\n      }\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.Subject',\n      value: (certificate.getSubject().str).replace(/\\//g, '\\n')\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.PublicKey',\n      value: certificate.getPublicKeyHex()\n    });\n    certificateAttributes.push({\n      translationKey: 'SignatureInfo.PublicKeyParameters',\n      value: certificate.getPublicKeyIdx()\n    });\n    return certificateAttributes;\n  }\n\n  getGeneralAttributes(certificate): SignatureAttribute[] {\n    let attributes: {\n      translationKey: string,\n      value?: any,\n      validation?: any\n    }[] = [];\n    attributes.push({\n      translationKey: 'SignatureInfo.IssuedTo',\n      // get certificate subject common name\n      value: (certificate.getSubjectString()).split('/').filter(x => x.includes('CN')).join(',').split('=')[1]\n    });\n    attributes.push({\n      translationKey: 'SignatureInfo.Issuer',\n      // get certificate issuer common name (CN)\n      value: (certificate.getIssuerString()).split('/').filter(x => x.includes('CN')).join(',').split('=')[1]\n    });\n    const notBefore = this._certificateService.parseCertificateDate(\n      certificate.getNotBefore()\n    );\n    attributes.push({\n      translationKey: 'SignatureInfo.NotValidBeforeShort',\n      value: notBefore,\n      validation: {\n        isValid: notBefore ? new Date() >= notBefore : false,\n        showValidation: true\n      }\n    });\n    const notAfter = this._certificateService.parseCertificateDate(\n      certificate.getNotAfter()\n    );\n    attributes.push({\n      translationKey: 'SignatureInfo.NotValidAfterShort',\n      value:  notAfter,\n      validation: {\n        isValid: notAfter ? notAfter >= new Date() : false,\n        showValidation: true\n      }\n    });\n    return attributes;\n  }\n\n  hideCertificateModal() {\n    this._bsModalRef.hide();\n  }\n}\n","<div class=\"modal-header p-0\">\n  <button\n    (click)=\"hideCertificateModal()\"\n    type=\"button\"\n    class=\"close\"\n    data-dismiss=\"modal\"\n    aria-label=\"Cancel\"\n  >\n    <span aria-hidden=\"true\">&times;</span>\n  </button>\n</div>\n<div class=\"modal-body mt-0 pt-0\">\n  <div class=\"card-body\" *ngIf=\"user\">\n    <div class=\"row\">\n      <div class=\"col-12\">\n        <h5 class=\"mt-2 mb-3\">\n          {{user?.familyName}} {{user?.givenName}}\n          <span *ngIf=\"dateCreated\">{{ dateCreated | date:'short' }}</span>\n          (\n          <span *ngIf=\"isVerified\" class=\"text-purple\" [translate]=\"'SignatureInfo.Verified'\"></span>\n          <span *ngIf=\"!isVerified\" class=\"text-red\" [translate]=\"'SignatureInfo.NotVerified'\"></span>\n          )\n          <small> {{examinationCode}} </small>\n        </h5>\n      </div>\n    </div>\n    <div class=\"mb-3\">\n      <div class=\" d-none d-md-block\">\n        <ul class=\"nav nav-tabs\">\n          <li class=\"nav-item\">\n            <a class=\"nav-link active\" [class.active]=\"tab === 1\" rel=\"tab1\"\n               (click)=\"tab =1\">{{'SignatureInfo.General' | translate}}</a>\n          </li>\n          <li class=\"nav-item\">\n            <a class=\"nav-link\" [class.active]=\"tab === 2\" rel=\"tab2\"\n               (click)=\"tab =2\"> {{'SignatureInfo.Details' | translate}}</a>\n          </li>\n        </ul>\n      </div>\n    </div>\n\n    <div *ngIf=\"tab===1\">\n      <div class=\"card\">\n        <h5 class=\"d-block card-header mt-2 mb-n2 align-items-center\">\n          <i class=\"fa fa-certificate\"></i>\n          {{'SignatureInfo.CertificateInformation' | translate}}\n        </h5>\n        <hr/>\n        <div class=\"d-block card-body\">\n          <div class=\"container font-lg mt-n2 pt-n2\" *ngIf=\"purposes && purposes.length\">\n            <div class=\"row\">\n              <div>\n                <span> {{'SignatureInfo.CertificatePurposes.Header' |translate}}</span>\n              </div>\n            </div>\n            <ul>\n              <li *ngFor=\"let purpose of purposes\">\n                {{'SignatureInfo.CertificatePurposes.' + purpose |translate}}\n              </li>\n            </ul>\n            <hr/>\n          </div>\n          <div *ngFor=\"let attribute of generalAttributes | slice: 0:2\" class=\"row font-lg\">\n            <div *ngIf=\"attribute && attribute.value\">\n              <div class=\"flex-sm-column flex-wrap\">\n                <span class=\"text-theme col-4 text-nowrap\">{{attribute.translationKey | translate}}: </span>\n                <span class=\"col-8\">{{attribute.value}}</span>\n              </div>\n            </div>\n          </div>\n          <div class=\"row font-lg\">\n          <span *ngFor=\"let attribute of generalAttributes |slice: 2:4\" class=\"flex-sm-column\">\n            <span *ngIf=\"attribute && attribute.value\">\n              <span class=\"col-4 text-theme text-nowrap\">{{attribute.translationKey | translate}}: </span>\n              <span class=\"col-4 pr-2\">{{attribute.value |date:'short'}}</span>\n            </span>\n          </span>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <div *ngIf=\"tab=== 2\">\n      <div id=\"group-list\" class=\"flex-column\">\n        <div class=\"sis--list-group sis--list-group-body\">\n          <ul class=\"sis--list-group sis--list-group-header\">\n            <li class=\"sis--list-group-item\">\n              <div class=\"d-flex justify-content-center align-items-center\">\n                <div class=\"sis--list-group-item_lg_column d-none d-block\"\n                     [translate]=\"'SignatureInfo.Field'\"></div>\n                <div class=\"sis--list-group-item_lg_column d-none d-block px-4\"\n                     [translate]=\"'SignatureInfo.Value'\"></div>\n              </div>\n            </li>\n          </ul>\n          <ul class=\"sis--list-group sis--list-group-body\">\n            <li class=\"sis--list-group-item\" *ngFor=\"let attribute of signatureAttributes\">\n              <div class=\"d-flex justify-content-center align-items-center list-color\">\n                <div class=\"sis--list-group-item_lg_column d-none d-block flex-wrap\">\n              <span class=\"text-secondary\">\n                <i class=\"fa fa-certificate\"></i>\n                {{attribute.translationKey | translate}}\n              </span>\n                </div>\n                <div class=\"sis--list-group-item_lg_column d-block flex-wrap\">\n              <span class=\"text-secondary p-2 p-md-0 text-break \" [ngClass]=\"{\n                'text-red': attribute.validation && attribute.validation.showValidation && !attribute.validation.isValid,\n                'text-green': attribute.validation && attribute.validation.showValidation && attribute.validation.isValid\n              }\">{{ attribute.value}}</span>\n                </div>\n              </div>\n            </li>\n          </ul>\n        </div>\n      </div>\n    </div>\n  </div>\n  <div class=\"row\">\n    <div class=\"col text-center\">\n      <button\n        (click)=\"hideCertificateModal()\"\n        class=\"btn btn-success\"\n        [translate]=\"'SignatureInfo.Close'\"\n      ></button>\n    </div>\n  </div>\n</div>\n"]}