@colijnit/sharedcomponents 259.1.7 → 259.1.8
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/358.645812766f7a400d0d38.js +1 -0
- package/3rdpartylicenses.txt +2017 -0
- package/719.eae3283e9de629a67fcc.js +1 -0
- package/977.bd6291f9ee6f6ddf91f1.js +1 -0
- package/bundles/colijnit-sharedcomponents.umd.js +121 -54
- 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 +112 -48
- package/favicon.ico +0 -0
- package/fesm2015/colijnit-sharedcomponents.js +111 -47
- package/fesm2015/colijnit-sharedcomponents.js.map +1 -1
- package/index.html +12 -0
- package/lib/components/custom-pdf/custom-pdf-dialog.component.d.ts +3 -0
- package/main.987141ee790ebcd9f915.js +1 -0
- package/package.json +1 -1
- package/polyfills.907fe9d1887c5de17993.js +1 -0
- package/runtime.75ebd9b0a664565e2d37.js +1 -0
- package/styles.d0f69b6bddb7c4e48842.css +1 -0
|
@@ -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,7 +245,7 @@ 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
|
<section [innerHTML]="fileBody"></section>
|
|
187
251
|
</div>
|
|
@@ -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        <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  // 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"]}
|
package/favicon.ico
ADDED
|
Binary file
|
|
@@ -3488,22 +3488,17 @@ class CustomPdfDialogComponent {
|
|
|
3488
3488
|
this.additionalFileChangeEvent = new EventEmitter();
|
|
3489
3489
|
this.signaturePads = {};
|
|
3490
3490
|
this.signatureCanvases = {};
|
|
3491
|
+
this.fileBody = '';
|
|
3492
|
+
this.fileStyle = '';
|
|
3491
3493
|
this.showLoader = false;
|
|
3494
|
+
// set to true if you want to skip email for debug purposes
|
|
3495
|
+
this.enableLocalPreview = false;
|
|
3492
3496
|
}
|
|
3493
3497
|
showClass() {
|
|
3494
3498
|
return true;
|
|
3495
3499
|
}
|
|
3496
3500
|
ngOnInit() {
|
|
3497
|
-
|
|
3498
|
-
const head = document.getElementsByTagName('head')[0];
|
|
3499
|
-
const styles = this.extractStyleParts(this.additionalFileContents);
|
|
3500
|
-
this.fileStyle = styles[0];
|
|
3501
|
-
const style = document.createElement('style');
|
|
3502
|
-
style.appendChild(document.createTextNode(this.fileStyle));
|
|
3503
|
-
head.appendChild(style);
|
|
3504
|
-
const body = this.extractBodyContents(this.additionalFileContents);
|
|
3505
|
-
this.fileBody = this._sanitizer.bypassSecurityTrustHtml(body[0]);
|
|
3506
|
-
}
|
|
3501
|
+
this._processAdditionalFileContents();
|
|
3507
3502
|
}
|
|
3508
3503
|
ngAfterViewInit() {
|
|
3509
3504
|
['signatureCanvas', 'signatureCanvas2'].forEach(id => {
|
|
@@ -3545,75 +3540,144 @@ class CustomPdfDialogComponent {
|
|
|
3545
3540
|
}
|
|
3546
3541
|
});
|
|
3547
3542
|
}
|
|
3543
|
+
_processAdditionalFileContents() {
|
|
3544
|
+
var _a;
|
|
3545
|
+
if (!this.additionalFileContents)
|
|
3546
|
+
return;
|
|
3547
|
+
const styles = this.extractStyleParts(this.additionalFileContents);
|
|
3548
|
+
this.fileStyle = styles.join('\n').trim();
|
|
3549
|
+
if (this.fileStyle) {
|
|
3550
|
+
const head = document.getElementsByTagName('head')[0];
|
|
3551
|
+
const styleEl = document.createElement('style');
|
|
3552
|
+
styleEl.appendChild(document.createTextNode(this.fileStyle));
|
|
3553
|
+
head.appendChild(styleEl);
|
|
3554
|
+
}
|
|
3555
|
+
const bodyParts = this.extractBodyContents(this.additionalFileContents);
|
|
3556
|
+
const bodyHtml = ((_a = bodyParts[0]) !== null && _a !== void 0 ? _a : this.removeStyleAndScriptTags(this.additionalFileContents)).trim();
|
|
3557
|
+
this.fileBody = this._sanitizer.bypassSecurityTrustHtml(bodyHtml);
|
|
3558
|
+
}
|
|
3548
3559
|
extractStyleParts(html) {
|
|
3560
|
+
var _a;
|
|
3549
3561
|
const stylePattern = /<style[^>]*>([\s\S]*?)<\/style>/gi;
|
|
3550
3562
|
const matches = [];
|
|
3551
3563
|
let match;
|
|
3552
3564
|
while ((match = stylePattern.exec(html)) !== null) {
|
|
3553
|
-
matches.push(match[1].trim());
|
|
3565
|
+
matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
|
|
3554
3566
|
}
|
|
3555
3567
|
return matches;
|
|
3556
3568
|
}
|
|
3557
3569
|
removeStyleAndScriptTags(html) {
|
|
3558
3570
|
const styleTagPattern = /<style[^>]*>[\s\S]*?<\/style>/gi;
|
|
3559
3571
|
const scriptTagPattern = /<script[^>]*>[\s\S]*?<\/script>/gi;
|
|
3560
|
-
return html.replace(styleTagPattern, '').replace(scriptTagPattern, '');
|
|
3572
|
+
return (html || '').replace(styleTagPattern, '').replace(scriptTagPattern, '');
|
|
3561
3573
|
}
|
|
3562
3574
|
extractBodyContents(html) {
|
|
3575
|
+
var _a;
|
|
3563
3576
|
html = this.removeStyleAndScriptTags(html);
|
|
3564
|
-
const
|
|
3577
|
+
const bodyPattern = /<body[^>]*>([\s\S]*?)<\/body>/gi;
|
|
3565
3578
|
const matches = [];
|
|
3566
3579
|
let match;
|
|
3567
|
-
while ((match =
|
|
3568
|
-
matches.push(match[1].trim());
|
|
3580
|
+
while ((match = bodyPattern.exec(html)) !== null) {
|
|
3581
|
+
matches.push(((_a = match[1]) !== null && _a !== void 0 ? _a : '').trim());
|
|
3569
3582
|
}
|
|
3570
3583
|
return matches;
|
|
3571
3584
|
}
|
|
3585
|
+
_isLocalhost() {
|
|
3586
|
+
return ['localhost', '127.0.0.1', '::1'].includes(location.hostname);
|
|
3587
|
+
}
|
|
3572
3588
|
handleSaveClicked() {
|
|
3589
|
+
var _a, _b;
|
|
3573
3590
|
return __awaiter(this, void 0, void 0, function* () {
|
|
3574
3591
|
this.showLoader = true;
|
|
3575
3592
|
const clearButton = document.getElementById('clearButton');
|
|
3576
|
-
clearButton
|
|
3577
|
-
|
|
3578
|
-
const
|
|
3579
|
-
|
|
3580
|
-
|
|
3581
|
-
|
|
3582
|
-
|
|
3593
|
+
if (clearButton)
|
|
3594
|
+
clearButton.hidden = true;
|
|
3595
|
+
const doc = new jsPDF({ unit: 'mm', format: 'a4' });
|
|
3596
|
+
let createdTemp = false;
|
|
3597
|
+
let root = ((_a = this.pdfBody) === null || _a === void 0 ? void 0 : _a.nativeElement) ||
|
|
3598
|
+
document.querySelector('#pdfBody');
|
|
3599
|
+
if (!root) {
|
|
3600
|
+
root = document.createElement('div');
|
|
3601
|
+
root.id = 'pdfBody';
|
|
3602
|
+
root.style.position = 'fixed';
|
|
3603
|
+
root.style.left = '-99999px';
|
|
3604
|
+
root.style.top = '0';
|
|
3605
|
+
root.style.width = '794px';
|
|
3606
|
+
root.style.pointerEvents = 'none';
|
|
3607
|
+
document.body.appendChild(root);
|
|
3608
|
+
createdTemp = true;
|
|
3609
|
+
}
|
|
3610
|
+
if (this.additionalFileContents && !root.hasChildNodes()) {
|
|
3611
|
+
const html = String(this.additionalFileContents);
|
|
3612
|
+
if (html.includes('<html')) {
|
|
3613
|
+
const parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
3614
|
+
root.innerHTML = ((_b = parsed.body) === null || _b === void 0 ? void 0 : _b.innerHTML) || html;
|
|
3615
|
+
}
|
|
3616
|
+
else {
|
|
3617
|
+
root.innerHTML = html;
|
|
3618
|
+
}
|
|
3619
|
+
}
|
|
3620
|
+
const allSections = Array.from(root.querySelectorAll('section'));
|
|
3621
|
+
const leafSections = allSections.filter(s => !s.querySelector('section'));
|
|
3622
|
+
const nodesToRender = leafSections.length ? leafSections : (allSections.length ? allSections : [root]);
|
|
3623
|
+
const pageWidth = doc.internal.pageSize.getWidth();
|
|
3624
|
+
const pageHeight = doc.internal.pageSize.getHeight();
|
|
3625
|
+
for (let i = 0; i < nodesToRender.length; i++) {
|
|
3626
|
+
const el = nodesToRender[i];
|
|
3583
3627
|
try {
|
|
3584
|
-
const dataUrl = yield toJpeg(
|
|
3585
|
-
quality: 1,
|
|
3586
|
-
backgroundColor: 'white'
|
|
3587
|
-
});
|
|
3588
|
-
const img = new Image();
|
|
3589
|
-
img.src = dataUrl;
|
|
3628
|
+
const dataUrl = yield toJpeg(el, { quality: 1, backgroundColor: 'white' });
|
|
3590
3629
|
yield new Promise((resolve) => {
|
|
3630
|
+
const img = new Image();
|
|
3591
3631
|
img.onload = () => {
|
|
3592
|
-
const
|
|
3593
|
-
const
|
|
3594
|
-
|
|
3595
|
-
|
|
3632
|
+
const w = img.width;
|
|
3633
|
+
const h = img.height;
|
|
3634
|
+
const scale = Math.min(pageWidth / w, pageHeight / h);
|
|
3635
|
+
const renderW = w * scale;
|
|
3636
|
+
const renderH = h * scale;
|
|
3637
|
+
doc.addImage(dataUrl, 'JPG', 0, 0, renderW, renderH);
|
|
3638
|
+
if (i < nodesToRender.length - 1)
|
|
3596
3639
|
doc.addPage();
|
|
3597
|
-
}
|
|
3598
3640
|
resolve();
|
|
3599
3641
|
};
|
|
3642
|
+
img.src = dataUrl;
|
|
3600
3643
|
});
|
|
3601
3644
|
}
|
|
3602
|
-
catch (
|
|
3603
|
-
console.error('Error generating image:',
|
|
3645
|
+
catch (err) {
|
|
3646
|
+
console.error('Error generating image:', err);
|
|
3604
3647
|
}
|
|
3605
3648
|
}
|
|
3606
|
-
|
|
3607
|
-
|
|
3608
|
-
|
|
3609
|
-
|
|
3610
|
-
|
|
3611
|
-
|
|
3612
|
-
|
|
3613
|
-
|
|
3614
|
-
|
|
3615
|
-
|
|
3616
|
-
|
|
3649
|
+
if (createdTemp && root && root.parentNode) {
|
|
3650
|
+
root.parentNode.removeChild(root);
|
|
3651
|
+
}
|
|
3652
|
+
// enableLocalPreview can be enabled to see the output without sending email
|
|
3653
|
+
if (this.enableLocalPreview && this._isLocalhost()) {
|
|
3654
|
+
const blob = doc.output('blob');
|
|
3655
|
+
const url = URL.createObjectURL(blob);
|
|
3656
|
+
window.open(url, '_blank');
|
|
3657
|
+
this.showLoader = false;
|
|
3658
|
+
if (clearButton)
|
|
3659
|
+
clearButton.hidden = false;
|
|
3660
|
+
return;
|
|
3661
|
+
}
|
|
3662
|
+
try {
|
|
3663
|
+
const tempFile = doc.output('datauristring');
|
|
3664
|
+
const file = this.dataURItoBlob(tempFile);
|
|
3665
|
+
const fileAsCoDocument = yield FileUtils.ReadFileAsNewCoDocument(file);
|
|
3666
|
+
fileAsCoDocument.fileType = FileUtils.IsImageFile(file) ? FileType$1.Image : FileType$1.Document;
|
|
3667
|
+
fileAsCoDocument.creationDate = new Date();
|
|
3668
|
+
fileAsCoDocument.modifiedDate = new Date();
|
|
3669
|
+
fileAsCoDocument.reports = 1;
|
|
3670
|
+
this.additionalFileChangeEvent.emit(fileAsCoDocument);
|
|
3671
|
+
}
|
|
3672
|
+
catch (e) {
|
|
3673
|
+
console.error('PDF packaging failed:', e);
|
|
3674
|
+
}
|
|
3675
|
+
finally {
|
|
3676
|
+
this.showLoader = false;
|
|
3677
|
+
this.closePDFDialog.emit();
|
|
3678
|
+
if (clearButton)
|
|
3679
|
+
clearButton.hidden = false;
|
|
3680
|
+
}
|
|
3617
3681
|
});
|
|
3618
3682
|
}
|
|
3619
3683
|
dataURItoBlob(dataURI) {
|
|
@@ -3651,7 +3715,7 @@ CustomPdfDialogComponent.decorators = [
|
|
|
3651
3715
|
<div *ngIf="showLoader" class="loader-container">
|
|
3652
3716
|
<co-loader *ngIf="showLoader"></co-loader>
|
|
3653
3717
|
</div>
|
|
3654
|
-
|
|
3718
|
+
|
|
3655
3719
|
<div #pdfBody id="pdfBody">
|
|
3656
3720
|
<section [innerHTML]="fileBody"></section>
|
|
3657
3721
|
</div>
|