@colijnit/sharedcomponents 259.1.7 → 259.1.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/colijnit-sharedcomponents.umd.js +123 -56
- package/bundles/colijnit-sharedcomponents.umd.js.map +1 -1
- package/colijnit-sharedcomponents.metadata.json +1 -1
- package/esm2015/lib/components/custom-pdf/custom-pdf-dialog.component.js +113 -49
- package/esm2015/lib/res/dictionary/dictionaries.js +3 -3
- package/fesm2015/colijnit-sharedcomponents.js +114 -50
- package/fesm2015/colijnit-sharedcomponents.js.map +1 -1
- package/lib/components/custom-pdf/custom-pdf-dialog.component.d.ts +3 -0
- package/lib/res/dictionary/dictionaries.d.ts +13 -0
- package/package.json +1 -1
|
@@ -18,22 +18,17 @@ export class CustomPdfDialogComponent {
|
|
|
18
18
|
this.additionalFileChangeEvent = new EventEmitter();
|
|
19
19
|
this.signaturePads = {};
|
|
20
20
|
this.signatureCanvases = {};
|
|
21
|
+
this.fileBody = '';
|
|
22
|
+
this.fileStyle = '';
|
|
21
23
|
this.showLoader = false;
|
|
24
|
+
// set to true if you want to skip email for debug purposes
|
|
25
|
+
this.enableLocalPreview = false;
|
|
22
26
|
}
|
|
23
27
|
showClass() {
|
|
24
28
|
return true;
|
|
25
29
|
}
|
|
26
30
|
ngOnInit() {
|
|
27
|
-
|
|
28
|
-
const head = document.getElementsByTagName('head')[0];
|
|
29
|
-
const styles = this.extractStyleParts(this.additionalFileContents);
|
|
30
|
-
this.fileStyle = styles[0];
|
|
31
|
-
const style = document.createElement('style');
|
|
32
|
-
style.appendChild(document.createTextNode(this.fileStyle));
|
|
33
|
-
head.appendChild(style);
|
|
34
|
-
const body = this.extractBodyContents(this.additionalFileContents);
|
|
35
|
-
this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);
|
|
36
|
-
}
|
|
31
|
+
this._processAdditionalFileContents();
|
|
37
32
|
}
|
|
38
33
|
ngAfterViewInit() {
|
|
39
34
|
['signatureCanvas', 'signatureCanvas2'].forEach(id => {
|
|
@@ -75,75 +70,144 @@ export class CustomPdfDialogComponent {
|
|
|
75
70
|
}
|
|
76
71
|
});
|
|
77
72
|
}
|
|
73
|
+
_processAdditionalFileContents() {
|
|
74
|
+
var _a;
|
|
75
|
+
if (!this.additionalFileContents)
|
|
76
|
+
return;
|
|
77
|
+
const styles = this.extractStyleParts(this.additionalFileContents);
|
|
78
|
+
this.fileStyle = styles.join('\n').trim();
|
|
79
|
+
if (this.fileStyle) {
|
|
80
|
+
const head = document.getElementsByTagName('head')[0];
|
|
81
|
+
const styleEl = document.createElement('style');
|
|
82
|
+
styleEl.appendChild(document.createTextNode(this.fileStyle));
|
|
83
|
+
head.appendChild(styleEl);
|
|
84
|
+
}
|
|
85
|
+
const bodyParts = this.extractBodyContents(this.additionalFileContents);
|
|
86
|
+
const bodyHtml = ((_a = bodyParts[0]) !== null && _a !== void 0 ? _a : this.removeStyleAndScriptTags(this.additionalFileContents)).trim();
|
|
87
|
+
this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);
|
|
88
|
+
}
|
|
78
89
|
extractStyleParts(html) {
|
|
90
|
+
var _a;
|
|
79
91
|
const stylePattern = /<style[^>]*>([\s\S]*?)<\/style>/gi;
|
|
80
92
|
const matches = [];
|
|
81
93
|
let match;
|
|
82
94
|
while ((match = stylePattern.exec(html)) !== null) {
|
|
83
|
-
matches.push(match[1].trim());
|
|
95
|
+
matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
|
|
84
96
|
}
|
|
85
97
|
return matches;
|
|
86
98
|
}
|
|
87
99
|
removeStyleAndScriptTags(html) {
|
|
88
100
|
const styleTagPattern = /<style[^>]*>[\s\S]*?<\/style>/gi;
|
|
89
101
|
const scriptTagPattern = /<script[^>]*>[\s\S]*?<\/script>/gi;
|
|
90
|
-
return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');
|
|
102
|
+
return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');
|
|
91
103
|
}
|
|
92
104
|
extractBodyContents(html) {
|
|
105
|
+
var _a;
|
|
93
106
|
html = this.removeStyleAndScriptTags(html);
|
|
94
|
-
const
|
|
107
|
+
const bodyPattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
|
|
95
108
|
const matches = [];
|
|
96
109
|
let match;
|
|
97
|
-
while ((match =
|
|
98
|
-
matches.push(match[1].trim());
|
|
110
|
+
while ((match = bodyPattern.exec(html)) !== null) {
|
|
111
|
+
matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
|
|
99
112
|
}
|
|
100
113
|
return matches;
|
|
101
114
|
}
|
|
115
|
+
_isLocalhost() {
|
|
116
|
+
return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);
|
|
117
|
+
}
|
|
102
118
|
handleSaveClicked() {
|
|
119
|
+
var _a, _b;
|
|
103
120
|
return __awaiter(this, void 0, void 0, function* () {
|
|
104
121
|
this.showLoader = true;
|
|
105
122
|
const clearButton = document.getElementById('clearButton');
|
|
106
|
-
clearButton
|
|
107
|
-
|
|
108
|
-
const
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
123
|
+
if (clearButton)
|
|
124
|
+
clearButton.hidden = true;
|
|
125
|
+
const doc = new jsPDF({ unit: 'mm', format: 'a4' });
|
|
126
|
+
let createdTemp = false;
|
|
127
|
+
let root = ((_a = this.pdfBody) === null || _a === void 0 ? void 0 : _a.nativeElement) ||
|
|
128
|
+
document.querySelector('#pdfBody');
|
|
129
|
+
if (!root) {
|
|
130
|
+
root = document.createElement('div');
|
|
131
|
+
root.id = 'pdfBody';
|
|
132
|
+
root.style.position = 'fixed';
|
|
133
|
+
root.style.left = '-99999px';
|
|
134
|
+
root.style.top = '0';
|
|
135
|
+
root.style.width = '794px';
|
|
136
|
+
root.style.pointerEvents = 'none';
|
|
137
|
+
document.body.appendChild(root);
|
|
138
|
+
createdTemp = true;
|
|
139
|
+
}
|
|
140
|
+
if (this.additionalFileContents && !root.hasChildNodes()) {
|
|
141
|
+
const html = String(this.additionalFileContents);
|
|
142
|
+
if (html.includes('<html')) {
|
|
143
|
+
const parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
144
|
+
root.innerHTML = ((_b = parsed.body) === null || _b === void 0 ? void 0 : _b.innerHTML) || html;
|
|
145
|
+
}
|
|
146
|
+
else {
|
|
147
|
+
root.innerHTML = html;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
const allSections = Array.from(root.querySelectorAll('section'));
|
|
151
|
+
const leafSections = allSections.filter(s => !s.querySelector('section'));
|
|
152
|
+
const nodesToRender = leafSections.length ? leafSections : (allSections.length ? allSections : [root]);
|
|
153
|
+
const pageWidth = doc.internal.pageSize.getWidth();
|
|
154
|
+
const pageHeight = doc.internal.pageSize.getHeight();
|
|
155
|
+
for (let i = 0; i < nodesToRender.length; i++) {
|
|
156
|
+
const el = nodesToRender[i];
|
|
113
157
|
try {
|
|
114
|
-
const dataUrl = yield toJpeg(
|
|
115
|
-
quality: 1,
|
|
116
|
-
backgroundColor: 'white'
|
|
117
|
-
});
|
|
118
|
-
const img = new Image();
|
|
119
|
-
img.src = dataUrl;
|
|
158
|
+
const dataUrl = yield toJpeg(el, { quality: 1, backgroundColor: 'white' });
|
|
120
159
|
yield new Promise((resolve) => {
|
|
160
|
+
const img = new Image();
|
|
121
161
|
img.onload = () => {
|
|
122
|
-
const
|
|
123
|
-
const
|
|
124
|
-
|
|
125
|
-
|
|
162
|
+
const w = img.width;
|
|
163
|
+
const h = img.height;
|
|
164
|
+
const scale = Math.min(pageWidth / w, pageHeight / h);
|
|
165
|
+
const renderW = w * scale;
|
|
166
|
+
const renderH = h * scale;
|
|
167
|
+
doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);
|
|
168
|
+
if (i < nodesToRender.length - 1)
|
|
126
169
|
doc.addPage();
|
|
127
|
-
}
|
|
128
170
|
resolve();
|
|
129
171
|
};
|
|
172
|
+
img.src = dataUrl;
|
|
130
173
|
});
|
|
131
174
|
}
|
|
132
|
-
catch (
|
|
133
|
-
console.error('Error generating image:',
|
|
175
|
+
catch (err) {
|
|
176
|
+
console.error('Error generating image:', err);
|
|
134
177
|
}
|
|
135
178
|
}
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
179
|
+
if (createdTemp && root && root.parentNode) {
|
|
180
|
+
root.parentNode.removeChild(root);
|
|
181
|
+
}
|
|
182
|
+
// enableLocalPreview can be enabled to see the output without sending email
|
|
183
|
+
if (this.enableLocalPreview && this._isLocalhost()) {
|
|
184
|
+
const blob = doc.output('blob');
|
|
185
|
+
const url = URL.createObjectURL(blob);
|
|
186
|
+
window.open(url, '_blank');
|
|
187
|
+
this.showLoader = false;
|
|
188
|
+
if (clearButton)
|
|
189
|
+
clearButton.hidden = false;
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
try {
|
|
193
|
+
const tempFile = doc.output('datauristring');
|
|
194
|
+
const file = this.dataURItoBlob(tempFile);
|
|
195
|
+
const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
|
|
196
|
+
fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;
|
|
197
|
+
fileAsCoDocument.creationDate = new Date();
|
|
198
|
+
fileAsCoDocument.modifiedDate = new Date();
|
|
199
|
+
fileAsCoDocument.reports = 1;
|
|
200
|
+
this.additionalFileChangeEvent.emit(fileAsCoDocument);
|
|
201
|
+
}
|
|
202
|
+
catch (e) {
|
|
203
|
+
console.error('PDF packaging failed:', e);
|
|
204
|
+
}
|
|
205
|
+
finally {
|
|
206
|
+
this.showLoader = false;
|
|
207
|
+
this.closePDFDialog.emit();
|
|
208
|
+
if (clearButton)
|
|
209
|
+
clearButton.hidden = false;
|
|
210
|
+
}
|
|
147
211
|
});
|
|
148
212
|
}
|
|
149
213
|
dataURItoBlob(dataURI) {
|
|
@@ -181,9 +245,9 @@ CustomPdfDialogComponent.decorators = [
|
|
|
181
245
|
<div *ngIf="showLoader" class="loader-container">
|
|
182
246
|
<co-loader *ngIf="showLoader"></co-loader>
|
|
183
247
|
</div>
|
|
184
|
-
|
|
248
|
+
|
|
185
249
|
<div #pdfBody id="pdfBody">
|
|
186
|
-
<
|
|
250
|
+
<div [innerHTML]="fileBody"></div>
|
|
187
251
|
</div>
|
|
188
252
|
<ng-template #footerTemplate>
|
|
189
253
|
<div class="co-dialog-footer-button-wrapper">
|
|
@@ -212,4 +276,4 @@ CustomPdfDialogComponent.propDecorators = {
|
|
|
212
276
|
closePDFDialog: [{ type: Output }],
|
|
213
277
|
additionalFileChangeEvent: [{ type: Output }]
|
|
214
278
|
};
|
|
215
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-pdf-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/sharedcomponents/src/lib/components/custom-pdf/custom-pdf-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ElementRef, EventEmitter, HostBinding, Input, OnInit, Output, ViewChild, ViewEncapsulation} from '@angular/core';\r\nimport {jsPDF} from 'jspdf';\r\nimport {IconCacheService} from '../../service/icon-cache.service';\r\nimport {DomSanitizer} from '@angular/platform-browser';\r\nimport SignaturePad from 'signature_pad';\r\nimport {toJpeg} from 'html-to-image';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {FileUtils} from '@colijnit/mainapi/build/utils/file-utils';\r\n\r\nimport {FileType} from '@colijnit/mainapi/build/enum/file-type.enum';\r\nimport {Icon} from '../../enum/icon.enum';\r\n\r\n@Component({\r\n  selector: 'co-custom-pdf-dialog',\r\n  template: `\r\n    <co-dialog\r\n      [footerTemplate]=\"footerTemplate\"\r\n      id=\"custom-pdf-dialog\"\r\n      (closeClick)=\"closePDFDialog.emit()\">\r\n      <!-- create a nice container for loader so it is not random in the corner somewhere -->\r\n      <div *ngIf=\"showLoader\" class=\"loader-container\">\r\n        <co-loader *ngIf=\"showLoader\"></co-loader>\r\n      </div>\r\n      \r\n      <div #pdfBody id=\"pdfBody\">\r\n        <section [innerHTML]=\"fileBody\"></section>\r\n      </div>\r\n      <ng-template #footerTemplate>\r\n        <div class=\"co-dialog-footer-button-wrapper\">\r\n          <co-button class=\"save-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CheckDuotone)\"\r\n                     (click)=\"handleSaveClicked()\"></co-button>\r\n          <co-button class=\"close-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CrossSkinny)\"\r\n                     (click)=\"closePDFDialog.emit()\"></co-button>\r\n        </div>\r\n      </ng-template>\r\n    </co-dialog>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class CustomPdfDialogComponent implements OnInit, AfterViewInit {\r\n  public readonly icons: typeof Icon = Icon;\r\n\r\n  @HostBinding('class.co-custom-pdf-dialog')\r\n  public showClass() {\r\n    return true;\r\n  }\r\n\r\n  @ViewChild('pdfBody')\r\n  public pdfBody: ElementRef;\r\n\r\n  @Input()\r\n  public additionalFileContents: any;\r\n\r\n  @Input()\r\n  public additionalFileName: string = '';\r\n\r\n  @Output()\r\n  public closePDFDialog: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Output()\r\n  public additionalFileChangeEvent: EventEmitter<CoDocument> = new EventEmitter<CoDocument>();\r\n\r\n  public canvas: HTMLCanvasElement;\r\n  public context: CanvasRenderingContext2D;\r\n  public signaturePads: { [key: string]: SignaturePad } = {};\r\n  public signatureCanvases: { [key: string]: ElementRef } = {};\r\n  public fileBody: any;\r\n  public fileStyle: string;\r\n\r\n  public showLoader: boolean = false;\r\n\r\n  constructor(\r\n    public iconCacheService: IconCacheService,\r\n    private _sanitizer: DomSanitizer\r\n  ) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (this.additionalFileContents) {\r\n      const head: HTMLHeadElement = document.getElementsByTagName('head')[0];\r\n      const styles: string[] = this.extractStyleParts(this.additionalFileContents);\r\n      this.fileStyle = styles[0];\r\n      const style: HTMLStyleElement = document.createElement('style');\r\n      style.appendChild(document.createTextNode(this.fileStyle));\r\n      head.appendChild(style);\r\n      const body: string[] = this.extractBodyContents(this.additionalFileContents);\r\n      this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    ['signatureCanvas', 'signatureCanvas2'].forEach(id => {\r\n      const canvasElement = document.getElementById(id);\r\n      if (canvasElement) {\r\n        const canvasRef = new ElementRef(canvasElement);\r\n        const signaturePad = new SignaturePad(<HTMLCanvasElement>canvasRef.nativeElement);\r\n        this.signatureCanvases[id] = canvasRef;\r\n        this.signaturePads[id] = signaturePad;\r\n\r\n        setTimeout(() => {\r\n          this._resizeSignatureCanvas(signaturePad, canvasRef);\r\n        });\r\n      }\r\n    });\r\n    const clearButton = document.getElementById('clearButton');\r\n    clearButton?.addEventListener('click', () => this.handleClear());\r\n\r\n    // Due to a bug or known issue with html-to-image we need a bit of JS the fill in the checked value when selecting a checkbox.\r\n    // Handling the checkboxes and radiobuttons\r\n    const container: HTMLElement = this.pdfBody.nativeElement as HTMLElement;\r\n    container.addEventListener('change', (event: Event) => {\r\n      const target = event.target as HTMLElement;\r\n      if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'checkbox') {\r\n        const checkbox: HTMLInputElement = target as HTMLInputElement;\r\n        if (checkbox.checked) {\r\n          checkbox.setAttribute('checked', 'true');\r\n        } else {\r\n          checkbox.removeAttribute('checked');\r\n        }\r\n      } else if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'radio') {\r\n        const radio: HTMLInputElement = target as HTMLInputElement;\r\n        const groupName: string = radio.name;\r\n        const radios: NodeListOf<Element> = container.querySelectorAll(`input[type=\"radio\"][name=\"${groupName}\"]`);\r\n        radios.forEach((radio) => radio.removeAttribute('checked'));\r\n        if (radio.checked) {\r\n          radio.setAttribute('checked', 'true');\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  public extractStyleParts(html: string): string[] {\r\n    const stylePattern: RegExp = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push(match[1].trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public removeStyleAndScriptTags(html: string): string {\r\n    const styleTagPattern: RegExp = /<style[^>]*>[\\s\\S]*?<\\/style>/gi;\r\n    const scriptTagPattern: RegExp = /<script[^>]*>[\\s\\S]*?<\\/script>/gi;\r\n    return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');\r\n  }\r\n\r\n  public extractBodyContents(html: string): string [] {\r\n    html = this.removeStyleAndScriptTags(html);\r\n    const stylePattern: RegExp = /<body[^>]*>([\\s\\S]*?)<\\/body>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push(match[1].trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public async handleSaveClicked(): Promise<void> {\r\n    this.showLoader = true;\r\n    const clearButton: HTMLElement = document.getElementById('clearButton');\r\n    clearButton.hidden = true;\r\n    const doc: jsPDF = new jsPDF();\r\n    const htmlDocument: HTMLElement = document.querySelector(\"#pdfBody\");\r\n    const documentSection: HTMLCollectionOf<HTMLElement> = htmlDocument.getElementsByTagName('section');\r\n    const sectionsCount: number = documentSection.length;\r\n\r\n    for (let i: number = 0; i < sectionsCount; i++) {\r\n      const html: HTMLElement = documentSection[i];\r\n      try {\r\n        const dataUrl: string = await toJpeg(html, {\r\n          quality: 1,\r\n          backgroundColor: 'white'\r\n        });\r\n        const img: HTMLImageElement = new Image();\r\n        img.src = dataUrl;\r\n        await new Promise<void>((resolve) => {\r\n          img.onload = () => {\r\n            const imgWidth: number = 218;\r\n            const imgHeight: number = (img.height * imgWidth) / img.width;\r\n            doc.addImage(dataUrl, 'JPG', -4, 0, imgWidth, imgHeight);\r\n            if (i < sectionsCount - 1) {\r\n              doc.addPage();\r\n            }\r\n            resolve();\r\n          };\r\n        });\r\n      } catch (error) {\r\n        console.error('Error generating image:', error);\r\n      }\r\n    }\r\n    const tempFile: string = doc.output('datauristring');\r\n    const file: File = this.dataURItoBlob(tempFile);\r\n    const fileAsCoDocument = await FileUtils.ReadFileAsNewCoDocument(file);\r\n    fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;\r\n    fileAsCoDocument.creationDate = new Date();\r\n    fileAsCoDocument.modifiedDate = new Date();\r\n    fileAsCoDocument.reports = 1;\r\n    this.additionalFileChangeEvent.emit(fileAsCoDocument);\r\n    this.showLoader = false;\r\n    this.closePDFDialog.emit();\r\n    clearButton.hidden = false;\r\n  }\r\n\r\n  public dataURItoBlob(dataURI): File {\r\n    const byteString: string = atob(dataURI.split(',')[1]);\r\n    const ab: ArrayBuffer = new ArrayBuffer(byteString.length);\r\n    const ia: Uint8Array = new Uint8Array(ab);\r\n    for (let i: number = 0; i < byteString.length; i++) {\r\n      ia[i] = byteString.charCodeAt(i);\r\n    }\r\n    return new File([ab], this.additionalFileName + '.pdf');\r\n  }\r\n\r\n  public handleClear(): void {\r\n    Object.values(this.signaturePads).forEach(signaturePad => {\r\n      signaturePad?.clear();\r\n    });\r\n  }\r\n\r\n  private _resizeSignatureCanvas(signature: SignaturePad, signatureCanvas: ElementRef): void {\r\n    const imageBeforeResize: string = signature.toDataURL();\r\n    const ratio: number = Math.max(window.devicePixelRatio || 1, 1);\r\n    signatureCanvas.nativeElement.width = signatureCanvas.nativeElement.offsetWidth * ratio;\r\n    signatureCanvas.nativeElement.height = signatureCanvas.nativeElement.offsetHeight * ratio;\r\n    signatureCanvas.nativeElement.getContext(\"2d\").scale(ratio, ratio);\r\n    signature.fromDataURL(imageBeforeResize);\r\n  }\r\n}\r\n"]}
|
|
279
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-pdf-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/sharedcomponents/src/lib/components/custom-pdf/custom-pdf-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ElementRef, EventEmitter, HostBinding, Input, OnInit, Output, ViewChild, ViewEncapsulation} from '@angular/core';\r\nimport {jsPDF} from 'jspdf';\r\nimport {IconCacheService} from '../../service/icon-cache.service';\r\nimport {DomSanitizer} from '@angular/platform-browser';\r\nimport SignaturePad from 'signature_pad';\r\nimport {toJpeg} from 'html-to-image';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {FileUtils} from '@colijnit/mainapi/build/utils/file-utils';\r\n\r\nimport {FileType} from '@colijnit/mainapi/build/enum/file-type.enum';\r\nimport {Icon} from '../../enum/icon.enum';\r\n\r\n@Component({\r\n  selector: 'co-custom-pdf-dialog',\r\n  template: `\r\n    <co-dialog\r\n      [footerTemplate]=\"footerTemplate\"\r\n      id=\"custom-pdf-dialog\"\r\n      (closeClick)=\"closePDFDialog.emit()\">\r\n      <!-- create a nice container for loader so it is not random in the corner somewhere -->\r\n      <div *ngIf=\"showLoader\" class=\"loader-container\">\r\n        <co-loader *ngIf=\"showLoader\"></co-loader>\r\n      </div>\r\n\r\n      <div #pdfBody id=\"pdfBody\">\r\n        <div [innerHTML]=\"fileBody\"></div>\r\n      </div>\r\n      <ng-template #footerTemplate>\r\n        <div class=\"co-dialog-footer-button-wrapper\">\r\n          <co-button class=\"save-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CheckDuotone)\"\r\n                     (click)=\"handleSaveClicked()\"></co-button>\r\n          <co-button class=\"close-button\"\r\n                     [iconData]=\"iconCacheService.getIcon(icons.CrossSkinny)\"\r\n                     (click)=\"closePDFDialog.emit()\"></co-button>\r\n        </div>\r\n      </ng-template>\r\n    </co-dialog>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None\r\n})\r\nexport class CustomPdfDialogComponent implements OnInit, AfterViewInit {\r\n  public readonly icons: typeof Icon = Icon;\r\n\r\n  @HostBinding('class.co-custom-pdf-dialog')\r\n  public showClass() {\r\n    return true;\r\n  }\r\n\r\n  @ViewChild('pdfBody')\r\n  public pdfBody: ElementRef;\r\n\r\n  @Input()\r\n  public additionalFileContents: any;\r\n\r\n  @Input()\r\n  public additionalFileName: string = '';\r\n\r\n  @Output()\r\n  public closePDFDialog: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Output()\r\n  public additionalFileChangeEvent: EventEmitter<CoDocument> = new EventEmitter<CoDocument>();\r\n\r\n  public canvas: HTMLCanvasElement;\r\n  public context: CanvasRenderingContext2D;\r\n  public signaturePads: { [key: string]: SignaturePad } = {};\r\n  public signatureCanvases: { [key: string]: ElementRef } = {};\r\n  public fileBody: any = '';\r\n  public fileStyle: string = '';\r\n\r\n  public showLoader: boolean = false;\r\n\r\n  // set to true if you want to skip email for debug purposes\r\n  private enableLocalPreview = false;\r\n\r\n  constructor(\r\n    public iconCacheService: IconCacheService,\r\n    private _sanitizer: DomSanitizer\r\n  ) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this._processAdditionalFileContents();\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    ['signatureCanvas', 'signatureCanvas2'].forEach(id => {\r\n      const canvasElement = document.getElementById(id);\r\n      if (canvasElement) {\r\n        const canvasRef = new ElementRef(canvasElement);\r\n        const signaturePad = new SignaturePad(<HTMLCanvasElement>canvasRef.nativeElement);\r\n        this.signatureCanvases[id] = canvasRef;\r\n        this.signaturePads[id] = signaturePad;\r\n\r\n        setTimeout(() => {\r\n          this._resizeSignatureCanvas(signaturePad, canvasRef);\r\n        });\r\n      }\r\n    });\r\n    const clearButton = document.getElementById('clearButton');\r\n    clearButton?.addEventListener('click', () => this.handleClear());\r\n\r\n    // Due to a bug or known issue with html-to-image we need a bit of JS the fill in the checked value when selecting a checkbox.\r\n    // Handling the checkboxes and radiobuttons\r\n    const container: HTMLElement = this.pdfBody.nativeElement as HTMLElement;\r\n    container.addEventListener('change', (event: Event) => {\r\n      const target = event.target as HTMLElement;\r\n      if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'checkbox') {\r\n        const checkbox: HTMLInputElement = target as HTMLInputElement;\r\n        if (checkbox.checked) {\r\n          checkbox.setAttribute('checked', 'true');\r\n        } else {\r\n          checkbox.removeAttribute('checked');\r\n        }\r\n      } else if (target.tagName === 'INPUT' && (target as HTMLInputElement).type === 'radio') {\r\n        const radio: HTMLInputElement = target as HTMLInputElement;\r\n        const groupName: string = radio.name;\r\n        const radios: NodeListOf<Element> = container.querySelectorAll(`input[type=\"radio\"][name=\"${groupName}\"]`);\r\n        radios.forEach((radio) => radio.removeAttribute('checked'));\r\n        if (radio.checked) {\r\n          radio.setAttribute('checked', 'true');\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private _processAdditionalFileContents(): void {\r\n    if (!this.additionalFileContents) return;\r\n\r\n    const styles = this.extractStyleParts(this.additionalFileContents);\r\n    this.fileStyle = styles.join('\\n').trim();\r\n    if (this.fileStyle) {\r\n      const head = document.getElementsByTagName('head')[0];\r\n      const styleEl = document.createElement('style');\r\n      styleEl.appendChild(document.createTextNode(this.fileStyle));\r\n      head.appendChild(styleEl);\r\n    }\r\n\r\n    const bodyParts = this.extractBodyContents(this.additionalFileContents);\r\n    const bodyHtml = (bodyParts[0] ?? this.removeStyleAndScriptTags(this.additionalFileContents)).trim();\r\n    this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);\r\n  }\r\n\r\n  public extractStyleParts(html: string): string[] {\r\n    const stylePattern = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = stylePattern.exec(html)) !== null) {\r\n      matches.push((match[1] ?? '').trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  public removeStyleAndScriptTags(html: string): string {\r\n    const styleTagPattern = /<style[^>]*>[\\s\\S]*?<\\/style>/gi;\r\n    const scriptTagPattern = /<script[^>]*>[\\s\\S]*?<\\/script>/gi;\r\n    return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');\r\n  }\r\n\r\n  public extractBodyContents(html: string): string[] {\r\n    html = this.removeStyleAndScriptTags(html);\r\n    const bodyPattern = /<body[^>]*>([\\s\\S]*?)<\\/body>/gi;\r\n    const matches: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n    while ((match = bodyPattern.exec(html)) !== null) {\r\n      matches.push((match[1] ?? '').trim());\r\n    }\r\n    return matches;\r\n  }\r\n\r\n  private _isLocalhost(): boolean {\r\n    return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);\r\n  }\r\n\r\n  public async handleSaveClicked(): Promise<void> {\r\n    this.showLoader = true;\r\n\r\n    const clearButton = document.getElementById('clearButton');\r\n    if (clearButton) clearButton.hidden = true;\r\n\r\n    const doc = new jsPDF({ unit: 'mm', format: 'a4' });\r\n\r\n    let createdTemp = false;\r\n    let root: HTMLElement | null =\r\n      (this.pdfBody?.nativeElement as HTMLElement) ||\r\n      (document.querySelector('#pdfBody') as HTMLElement);\r\n\r\n    if (!root) {\r\n      root = document.createElement('div');\r\n      root.id = 'pdfBody';\r\n      root.style.position = 'fixed';\r\n      root.style.left = '-99999px';\r\n      root.style.top = '0';\r\n      root.style.width = '794px';\r\n      root.style.pointerEvents = 'none';\r\n      document.body.appendChild(root);\r\n      createdTemp = true;\r\n    }\r\n\r\n    if (this.additionalFileContents && !root.hasChildNodes()) {\r\n      const html = String(this.additionalFileContents);\r\n      if (html.includes('<html')) {\r\n        const parsed = new DOMParser().parseFromString(html, 'text/html');\r\n        root.innerHTML = parsed.body?.innerHTML || html;\r\n      } else {\r\n        root.innerHTML = html;\r\n      }\r\n    }\r\n\r\n    const allSections = Array.from(root.querySelectorAll('section'));\r\n    const leafSections = allSections.filter(s => !s.querySelector('section'));\r\n    const nodesToRender: HTMLElement[] =\r\n      leafSections.length ? leafSections : (allSections.length ? allSections : [root]);\r\n\r\n    const pageWidth = doc.internal.pageSize.getWidth();\r\n    const pageHeight = doc.internal.pageSize.getHeight();\r\n\r\n    for (let i = 0; i < nodesToRender.length; i++) {\r\n      const el = nodesToRender[i];\r\n      try {\r\n        const dataUrl = await toJpeg(el, { quality: 1, backgroundColor: 'white' });\r\n        await new Promise<void>((resolve) => {\r\n          const img = new Image();\r\n          img.onload = () => {\r\n            const w = img.width;\r\n            const h = img.height;\r\n            const scale = Math.min(pageWidth / w, pageHeight / h);\r\n            const renderW = w * scale;\r\n            const renderH = h * scale;\r\n\r\n            doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);\r\n            if (i < nodesToRender.length - 1) doc.addPage();\r\n            resolve();\r\n          };\r\n          img.src = dataUrl;\r\n        });\r\n      } catch (err) {\r\n        console.error('Error generating image:', err);\r\n      }\r\n    }\r\n\r\n    if (createdTemp && root && root.parentNode) {\r\n      root.parentNode.removeChild(root);\r\n    }\r\n\r\n    // enableLocalPreview can be enabled to see the output without sending email\r\n    if (this.enableLocalPreview && this._isLocalhost()) {\r\n      const blob = doc.output('blob');\r\n      const url = URL.createObjectURL(blob);\r\n      window.open(url as unknown as string, '_blank');\r\n      this.showLoader = false;\r\n      if (clearButton) clearButton.hidden = false;\r\n      return;\r\n    }\r\n\r\n    try {\r\n      const tempFile = doc.output('datauristring');\r\n      const file = this.dataURItoBlob(tempFile);\r\n      const fileAsCoDocument = await FileUtils.ReadFileAsNewCoDocument(file);\r\n      fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType.Image : FileType.Document;\r\n      fileAsCoDocument.creationDate = new Date();\r\n      fileAsCoDocument.modifiedDate = new Date();\r\n      fileAsCoDocument.reports = 1;\r\n      this.additionalFileChangeEvent.emit(fileAsCoDocument);\r\n    } catch (e) {\r\n      console.error('PDF packaging failed:', e);\r\n    } finally {\r\n      this.showLoader = false;\r\n      this.closePDFDialog.emit();\r\n      if (clearButton) clearButton.hidden = false;\r\n    }\r\n  }\r\n\r\n  public dataURItoBlob(dataURI): File {\r\n    const byteString: string = atob(dataURI.split(',')[1]);\r\n    const ab: ArrayBuffer = new ArrayBuffer(byteString.length);\r\n    const ia: Uint8Array = new Uint8Array(ab);\r\n    for (let i: number = 0; i < byteString.length; i++) {\r\n      ia[i] = byteString.charCodeAt(i);\r\n    }\r\n    return new File([ab], this.additionalFileName + '.pdf');\r\n  }\r\n\r\n  public handleClear(): void {\r\n    Object.values(this.signaturePads).forEach(signaturePad => {\r\n      signaturePad?.clear();\r\n    });\r\n  }\r\n\r\n  private _resizeSignatureCanvas(signature: SignaturePad, signatureCanvas: ElementRef): void {\r\n    const imageBeforeResize: string = signature.toDataURL();\r\n    const ratio: number = Math.max(window.devicePixelRatio || 1, 1);\r\n    signatureCanvas.nativeElement.width = signatureCanvas.nativeElement.offsetWidth * ratio;\r\n    signatureCanvas.nativeElement.height = signatureCanvas.nativeElement.offsetHeight * ratio;\r\n    signatureCanvas.nativeElement.getContext(\"2d\").scale(ratio, ratio);\r\n    signature.fromDataURL(imageBeforeResize);\r\n  }\r\n}\r\n"]}
|